java.io.Serializable淺析

 Java API中java.io.Serializable接口源碼:java

1 public interface Serializable {2 }

  類經過實現java.io.Serializable接口能夠啓用其序列化功能。未實現次接口的類沒法使其任何狀態序列化或反序列化。可序列化類的全部子類型自己都是可序列化的。序列化接口沒有方法或字段,僅用於標識可序列化的語義。web

  Java的"對象序列化"能讓你將一個實現了Serializable接口的對象轉換成byte流,這樣往後要用這個對象時候,你就能把這些byte數據恢復出來,並據此從新構建那個對象了。數據庫

  要想序列化對象,你必須先建立一個OutputStream,而後把它嵌進ObjectOutputStream。這時,你就能用writeObject()方法把對象寫入OutputStream了。數組

  writeObject()方法負責寫入特定類的對象的狀態,以便相應的 readObject()方法能夠還原它。經過調用 out.defaultWriteObject 能夠調用保存 Object 的字段的默認機制。該方法自己不須要涉及屬於其超類或子類的狀態。狀態是經過使用 writeObject 方法或使用 DataOutput 支持的用於基本數據類型的方法將各個字段寫入 ObjectOutputStream 來保存的。安全

  讀的時候,你得把InputStream嵌到ObjectInputStream裏面,而後再調用readObject()方法。不過這樣讀出來的,只是一個Object的reference,所以在用以前,還得先下傳。readObject() 方法負責從流中讀取並還原類字段。它能夠調用 in.defaultReadObject 來調用默認機制,以還原對象的非靜態和非瞬態字段。  defaultReadObject()方法使用流中的信息來分配流中經過當前對象中相應命名字段保存的對象的字段。這用於處理類發展後須要添加新字段的情形。該方法自己不須要涉及屬於其超類或子類的狀態。狀態是經過使用 writeObject 方法或使用 DataOutput 支持的用於基本數據類型的方法將各個字段寫入 ObjectOutputStream 來保存的。網絡

  在序列化時,有幾點要注意的:socket

  1:當一個對象被序列化時,只保存對象的非靜態成員變量(包括聲明爲private的變量),不能保存任何的成員方法和靜態的成員變量。this

  2:若是一個對象的成員變量是一個對象,那麼這個對象的數據成員也會被序列化。spa

  3:若是一個可序列化的對象包含對某個不可序列化的對象的引用,那麼整個序列化操做將會失敗,而且會拋出一個NotSerializableException。咱們能夠將這個引用標記爲transient,那麼對象仍然能夠序列化。code

 

  一、序列化是幹什麼的?

  簡單說就是爲了保存在內存中的各類對象的狀態,而且能夠把保存的對象狀態再讀出來。雖然你能夠用你本身的各類各樣的方法來保存Object States,可是Java給你提供一種應該比你本身好的保存對象狀態的機制,那就是序列化。

  二、什麼狀況下須要序列化

  a)當你想把的內存中的對象保存到一個文件中或者數據庫中時候;
  b)當你想用套接字在網絡上傳送對象的時候;
  c)當你想經過RMI傳輸對象的時候;

  三、當對一個對象實現序列化時,究竟發生了什麼?

  在沒有序列化前,每一個保存在堆(Heap)中的對象都有相應的狀態(state),即實例變量(instance ariable)好比:

1 Foo myFoo = new Foo();2 myFoo .setWidth(37);3 myFoo.setHeight(70);

  當經過下面的代碼序列化以後,MyFoo對象中的width和Height實例變量的值(37,70)都被保存到foo.ser文件中,這樣之後又能夠把它從文件中讀出來,從新在堆中建立原來的對象。固然保存時候不只僅是保存對象的實例變量的值,JVM還要保存一些小量信息,好比類的類型等以便恢復原來的對象。

1 FileOutputStream fs = new FileOutputStream("foo.ser");2 ObjectOutputStream os = new ObjectOutputStream(fs);3 os.writeObject(myFoo);

  四、實現序列化(保存到一個文件)的步驟

  a)Make a FileOutputStream

  java 代碼
  FileOutputStream fs = new FileOutputStream("foo.ser");

  b)Make a ObjectOutputStream

  java 代碼
  ObjectOutputStream os = new ObjectOutputStream(fs);

  c)write the object

  java 代碼
  os.writeObject(myObject1);
  os.writeObject(myObject2);
  os.writeObject(myObject3);

  d) close the ObjectOutputStream

  java 代碼
  os.close();

  五、舉例說明

 

複製代碼

 1 public class Box implements Serializable { 2     private static final long serialVersionUID = -3450064362986273896L; 3      4     private int width; 5     private int height; 6      7     public static void main(String[] args) { 8         Box myBox=new Box(); 9         myBox.setWidth(50);10         myBox.setHeight(30);11         try {12             FileOutputStream fs=new FileOutputStream("F:\\foo.ser");13             ObjectOutputStream os=new ObjectOutputStream(fs);14             os.writeObject(myBox);15             os.close();16             FileInputStream fi=new FileInputStream("F:\\foo.ser");17             ObjectInputStream oi=new ObjectInputStream(fi);18             Box box=(Box)oi.readObject();19             oi.close();20             System.out.println(box.height+","+box.width);21         } catch (Exception e) {22             e.printStackTrace();23         }24     }25     26     public int getWidth() {27         return width;28     }29     public void setWidth(int width) {30         this.width = width;31     }32     public int getHeight() {33         return height;34     }35     public void setHeight(int height) {36         this.height = height;37     }38 }

複製代碼

  六、相關注意事項

  a)當一個父類實現序列化,子類自動實現序列化,不須要顯式實現Serializable接口;
  b)當一個對象的實例變量引用其餘對象,序列化該對象時也把引用對象進行序列化;
  c)並不是全部的對象均可以序列化,至於爲何不能夠,有不少緣由了,好比:

  1.安全方面的緣由,好比一個對象擁有private,public等field,對於一個要傳輸的對象,好比寫到文件,或者進行rmi傳輸 等等,在序列化進行傳輸的過程當中,這個對象的private等域是不受保護的。
  2. 資源分配方面的緣由,好比socket,thread類,若是能夠序列化,進行傳輸或者保存,也沒法對他們進行從新的資源分配,並且,也是沒有必要這樣實現。

 

  serialVersionUID

  序列化運行時使用一個稱爲 serialVersionUID 的版本號與每一個可序列化類相關聯,該序列號在反序列化過程當中用於驗證序列化對象的發送者和接收者是否爲該對象加載了與序列化兼容的類。若是接收者加載的該對象的類的 serialVersionUID 與對應的發送者的類的版本號不一樣,則反序列化將會致使  InvalidClassException。可序列化類能夠經過聲明名爲 "serialVersionUID" 的字段(該字段必須是靜態 (static)、最終 (final) 的 long 型字段)顯式聲明其本身的 serialVersionUID:

 ANY-ACCESS-MODIFIER static final long serialVersionUID = 42L;

  若是可序列化類未顯式聲明 serialVersionUID,則序列化運行時將基於該類的各個方面計算該類的默認 serialVersionUID 值,如「Java(TM) 對象序列化規範」中所述。不過,強烈建議 全部可序列化類都顯式聲明 serialVersionUID 值,緣由是計算默認的 serialVersionUID 對類的詳細信息具備較高的敏感性,根據編譯器實現的不一樣可能千差萬別,這樣在反序列化過程當中可能會致使意外的 InvalidClassException。所以,爲保證 serialVersionUID 值跨不一樣 java 編譯器實現的一致性,序列化類必須聲明一個明確的 serialVersionUID 值。還強烈建議使用 private修飾符顯示聲明 serialVersionUID(若是可能),緣由是這種聲明僅應用於直接聲明類 -- serialVersionUID 字段做爲繼承成員沒有用處。數組類不能聲明一個明確的 serialVersionUID,所以它們老是具備默認的計算值,可是數組類沒有匹配 serialVersionUID 值的要求。 

相關文章
相關標籤/搜索