1. 程式人生 > >java中序列化之子類繼承父類序列化

java中序列化之子類繼承父類序列化

當一個父類實現Serializable介面後,他的子類都將自動的實現序列化。 

  以下驗證了這一點: 

package Serial;
import java.io.Serializable; 
public class SuperC implements Serializable {//父類實現了序列化 
 int supervalue; 
 public SuperC(int supervalue) { 
  this.supervalue = supervalue; 
 } 
 public String toString() { 
  return "supervalue: "+supervalue; 
 } 


public class SubC extends SuperC {//子類 
 int subvalue; 

 public SubC(int supervalue,int subvalue) { 
  super(supervalue); 
  this.subvalue=subvalue; 
 } 

 public String toString() { 
  return super.toString()+" sub: "+subvalue; 
 } 


public class Test1 { 

 public static void main(String [] args){ 
  SubC subc=new SubC(100,200); 
  FileInputStream in=null; 
  FileOutputStream out=null; 
  ObjectInputStream oin=null; 
  ObjectOutputStream oout=null; 
  try { 
   out = new FileOutputStream("Test1.txt");//子類序列化 
   oout = new ObjectOutputStream(out); 
   oout.writeObject(subc); 
   oout.close(); 
   oout=null; 

   in = new FileInputStream("Test1.txt"); 
   oin = new ObjectInputStream(in); 
   SubC subc2=(SubC)oin.readObject();//子類反序列化 
   System.out.println(subc2); 
  } catch (Exception ex){ 
   ex.printStackTrace(); 
  } finally{ 
  …此處省略 
 } 


  執行結果如下: 

supervalue: 100 sub: 200 
  可見子類成功的序列化/反序列化了。 

  怎管讓子類實現序列化看起來是一件很簡單的事情,但有的時候,往往我們不能夠讓父類實現Serializable介面,原因是有時候父類是抽象的(這並沒有關係),並且父類不能夠強制每個子類都擁有序列化的能力。換句話說父類設計的目的僅僅是為了被繼承。 

  要為一個沒有實現Serializable介面的父類,編寫一個能夠序列化的子類是一件很麻煩的事情。java docs中提到: 

“To allow subtypes of non-serializable classes to be serialized, the subtype may assume responsibility for saving and restoring the state of the supertype's public, protected, and (if accessible) package fields. The subtype may assume this responsibility only if the class it extends has an accessible no-arg constructor to initialize the class's state. It is an error to declare a class Serializable if this is not the case. The error will be detected at runtime. ” 
  也就是說,要為一個沒有實現Serializable介面的父類,編寫一個能夠序列化的子類要做兩件事情: 

  其一、父類要有一個無參的constructor; 

  其二、子類要負責序列化(反序列化)父類的域。 

  我們將SuperC的Serializable介面去掉,而給SubC加上Serializable介面。執行後產生錯誤: 

java.lang.Error: Unresolved compilation problem: 
Serializable cannot be resolved or is not a valid superinterface 
at Serial.SubC.<init>(SubC.java:15) 
at Serial.Test1.main(Test1.java:19) 
Exception in thread "main" 
  果真如docs中所說的一樣,父類缺少無參建構函式是不行的。

  接下來,按照docs中的建議我們改寫這個例子:

public abstract class SuperC { 
 int supervalue; 
 public SuperC(int supervalue) { 
  this.supervalue = supervalue; 
 }
 public SuperC(){}//增加一個無參的constructor 
  public String toString() { 
   return "supervalue: "+supervalue; 
  } 
 } 

 public class SubC extends SuperC implements Serializable { 
  int subvalue; 

  public SubC(int supervalue,int subvalue) { 
   super(supervalue); 
   this.subvalue=subvalue; 
  } 

  public String toString() { 
   return super.toString()+" sub: "+subvalue; 
  } 

  private void writeObject(java.io.ObjectOutputStream out) 
  throws IOException{ 
   out.defaultWriteObject();//先序列化物件 
   out.writeInt(supervalue);//再序列化父類的域 
  } 
  private void readObject(java.io.ObjectInputStream in) 
  throws IOException, ClassNotFoundException{ 
   in.defaultReadObject();//先反序列化物件 
   supervalue=in.readInt();//再反序列化父類的域 
  } 
}
  執行結果證明了這種方法是正確的。在此處我們用到了writeObject/ readObject方法,這對方法如果存在的話,序列化時就會被呼叫,以代替預設的行為(以後還要探討,先了解這麼多)。我們在序列化時,首先呼叫了ObjectOutputStream的defaultWriteObject,它使用預設的序列化行為,然後序列化父類的域;反序列化的時候也一樣。 

  歸納一下: 

  目的 行為 

  為一個實現Serializable介面的父類,編寫一個能夠序列化的子類 子類將自動的實現序列化 

  為一個沒有實現Serializable介面的父類,編寫一個能夠序列化的子類 1, 父類要有一個無參的constructor;2, 子類要先序列化自身,然後子類要負責序列化父類的域 

跟多參考:http://www.ibm.com/developerworks/cn/java/j-lo-serial/