JAVA總結---序列化的三種方式


序列化和反序列化
序列化:可以將對象轉化成一個字節序列,便於存儲。
反序列化:將序列化的字節序列還原
優點:可以實現對象的"持久性”, 所謂持久性就是指對象的生命周期不取決於程序。

序列化需要:
所需類:ObjectInputStream和ObjectOutputStream
方法: readObject()和writeObject();


序列化方式一: 實現Serializable接口(隱式序列化)
通過實現Serializable接口,這種是隱式序列化(不需要手動),這種是最簡單的序列化方式,會自動序列化所有非statictransient關鍵字修飾的成員變量。
 


   
   
  
  
          
  1. class Student implements Serializable{
  2. private String name;
  3. private int age;
  4. public static int QQ = 1234;
  5. private transient String address = "CHINA";
  6. Student(String name, int age ){
  7. this.name = name;
  8. this.age = age;
  9. }
  10. public String toString() {
  11. return "name: " + name + "\n"
  12. + "age: " + age + "\n"
  13. + "QQ: " + QQ + "\n"
  14. + "address: " + address;
  15. }
  16. public void SetAge(int age) {
  17. this.age = age;
  18. }
  19. }
  20. public class SerializableDemo {
  21. public static void main(String[] args) throws IOException, ClassNotFoundException {
  22. //創建可序列化對象
  23. System.out.println( "原來的對象:");
  24. Student stu = new Student( "Ming", 16);
  25. System.out.println(stu);
  26. //創建序列化輸出流
  27. ByteArrayOutputStream buff = new ByteArrayOutputStream();
  28. ObjectOutputStream out = new ObjectOutputStream(buff);
  29. //將序列化對象存入緩沖區
  30. out.writeObject(stu);
  31. //修改相關值
  32. Student.QQ = 6666; // 發現打印結果QQ的值被改變
  33. stu.SetAge( 18); //發現值沒有被改變
  34. //從緩沖區取回被序列化的對象
  35. ObjectInputStream in = new ObjectInputStream( new ByteArrayInputStream(buff.toByteArray()));
  36. Student newStu = (Student) in.readObject();
  37. System.out.println( "序列化后取出的對象:");
  38. System.out.println(newStu);
  39. }
  40. }

打印結果:
原來的對象:
name: Ming
age: 16
QQ: 1234
address: CHINA
序列化后取出的對象:
name: Ming
age: 16
QQ: 6666
address: null

發現address(被transient)和QQ(被static)也沒有被序列化,中途修改QQ的值是為了以防讀者誤會QQ被序列化了。因為序列化可以保存對象的狀態,但是QQ的值被改變了,說明沒有被序列化。static成員不屬於對象實例,可能被別的對象修改沒辦法序列化,序列化是序列對象。對於address被反序列化后由於沒有對應的引用,所以為null。而且Serializable不會調用構造方法。
PS:細心的可能發現序列化很誘人,可以保存對象的初始信息,在以后可以回到這個初始狀態。

序列化方式二:實現Externalizable接口。(顯式序列化)
Externalizable接口繼承自Serializable, 我們在實現該接口時,必須實現writeExternal()和readExternal()方法,而且只能通過手動進行序列化,並且兩個方法是自動調用的,因此,這個序列化過程是可控的,可以自己選擇哪些部分序列化
 


   
   
  
  
          
  1. public class Blip implements Externalizable{
  2. private int i ;
  3. private String s;
  4. public Blip() {}
  5. public Blip(String x, int a) {
  6. System.out.println( "Blip (String x, int a)");
  7. s = x;
  8. i = a;
  9. }
  10. public String toString() {
  11. return s+i;
  12. }
  13. @Override
  14. public void writeExternal(ObjectOutput out) throws IOException {
  15. // TODO Auto-generated method stub
  16. System.out.println( "Blip.writeExternal");
  17. out.writeObject(s);
  18. out.writeInt(i);
  19. //
  20. }
  21. @Override
  22. public void readExternal(ObjectInput in) throws IOException, ClassNotFoundException {
  23. // TODO Auto-generated method stub
  24. System.out.println( "Blip.readExternal");
  25. s = (String)in.readObject();
  26. i = in.readInt();
  27. }
  28. public static void main(String[] args) throws FileNotFoundException, IOException, ClassNotFoundException {
  29. System.out.println( "Constructing objects");
  30. Blip b = new Blip( "A Stirng", 47);
  31. System.out.println(b);
  32. ObjectOutputStream o = new ObjectOutputStream( new FileOutputStream( "F://Demo//file1.txt"));
  33. System.out.println( "保存對象");
  34. o.writeObject(b);
  35. o.close();
  36. //獲得對象
  37. System.out.println( "獲取對象");
  38. ObjectInputStream in = new ObjectInputStream( new FileInputStream( "F://Demo//file1.txt"));
  39. System.out.println( "Recovering b");
  40. b = (Blip)in.readObject();
  41. System.out.println(b);
  42. }
  43. }

打印結果為:
Constructing objects
Blip (String x, int a)
A Stirng47
保存對象
Blip.writeExternal
獲取對象
Recovering b
Blip.readExternal
A Stirng47

當注釋掉writeExternal和readExternal方法后打印結果為:
Constructing objects
Blip (String x, int a)
A Stirng47
保存對象
Blip.writeExternal
獲取對象
Recovering b
Blip.readExternal
null0

說明:Externalizable類會調用public的構造函數先初始化對象,在調用所保存的內容將對象還原。假如構造方法不是public則會出現運行時錯誤。

序列化方式三:實現Serializable接口+添加writeObject()和readObject()方法。(顯+隱序列化)

如果想將方式一和方式二的優點都用到的話,可以采用方式三, 先實現Serializable接口,並且添加writeObject()和readObject()方法。注意這里是添加,不是重寫或者覆蓋。但是添加的這兩個方法必須有相應的格式。

  • 1,方法必須要被private修飾                                ----->才能被調用
  • 2,第一行調用默認的defaultRead/WriteObject(); ----->隱式序列化非static和transient
  • 3,調用read/writeObject()將獲得的值賦給相應的值  --->顯式序列化

   
   
  
  
          
  1. public class SerDemo implements Serializable{
  2. public transient int age = 23;
  3. public String name ;
  4. public SerDemo(){
  5. System.out.println( "默認構造器。。。");
  6. }
  7. public SerDemo(String name) {
  8. this.name = name;
  9. }
  10. private void writeObject(ObjectOutputStream stream) throws IOException {
  11. stream.defaultWriteObject();
  12. stream.writeInt(age);
  13. }
  14. private void readObject(ObjectInputStream stream) throws ClassNotFoundException, IOException {
  15. stream.defaultReadObject();
  16. age = stream.readInt();
  17. }
  18. public String toString() {
  19. return "年齡" + age + " " + name;
  20. }
  21. public static void main(String[] args) throws IOException, ClassNotFoundException {
  22. SerDemo stu = new SerDemo( "Ming");
  23. ByteArrayOutputStream bout = new ByteArrayOutputStream();
  24. ObjectOutputStream out = new ObjectOutputStream(bout);
  25. out.writeObject(stu);
  26. ObjectInputStream in = new ObjectInputStream( new ByteArrayInputStream(bout.toByteArray()));
  27. SerDemo stu1 = (SerDemo) in.readObject();
  28. System.out.println(stu1);
  29. }
  30. }

打印結果為:
年齡23  Ming
注釋掉stream.writeInt(age)和age= stream.readInt()后:
年齡0  Ming
方式三結合了顯式和隱式序列化,Ming被正常序列化,由於age被trancient修飾,所以需要顯式序列化。

 

原文地址:https://blog.csdn.net/AHuqihua/article/details/81331316


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM