Java 序列化

Java 序列化

Java 提供了一種對象序列化的機制,該機制中,一個對象可以被表示爲一個字節序列,該字節序列包括該對象的數據、有關對象的類型的信息和存儲在對象中數據的類型。

將序列化對象寫入文件之後,可以從文件中讀取出來,並且對它進行反序列化,也就是說,對象的類型信息、對象的數據,還有對象中的數據類型可以用來在內存中新建對象。

整個過程都是 Java 虛擬機(JVM)獨立的,也就是說,在一個平臺上序列化的對象可以在另一個完全不同的平臺上反序列化該對象。

類 ObjectInputStream 和 ObjectOutputStream 是高層次的數據流,它們包含序列化和反序列化對象的方法。

ObjectOutputStream 類包含很多寫方法來寫各種數據類型,但是一個特別的方法例外:

publicfinalvoidwriteObject(Objectx)throwsIOException

上面的方法序列化一個對象,並將它發送到輸出流。相似的 ObjectInputStream 類包含如下反序列化一個對象的方法:

publicfinalObjectreadObject()throwsIOException,ClassNotFoundException

該方法從流中取出下一個對象,並將對象反序列化。它的返回值爲Object,因此,你需要將它轉換成合適的數據類型。

爲了演示序列化在Java中是怎樣工作的,我將使用之前教程中提到的Employee類,假設我們定義瞭如下的Employee類,該類實現了Serializable 接口。

Employee.java 文件代碼:

publicclassEmployeeimplementsjava.io.Serializable{publicStringname;publicStringaddress;publictransientintSSN;publicintnumber;publicvoidmailCheck(){System.out.println("Mailing a check to " +name +"" +address);}}

請注意,一個類的對象要想序列化成功,必須滿足兩個條件:

該類必須實現 java.io.Serializable 對象。

該類的所有屬性必須是可序列化的。如果有一個屬性不是可序列化的,則該屬性必須註明是短暫的。

如果你想知道一個 Java 標準類是否是可序列化的,請查看該類的文檔。檢驗一個類的實例是否能序列化十分簡單, 只需要查看該類有沒有實現 java.io.Serializable接口。


序列化對象

ObjectOutputStream 類用來序列化一個對象,如下的 SerializeDemo 例子實例化了一個 Employee 對象,並將該對象序列化到一個文件中。

該程序執行後,就創建了一個名爲 employee.ser 文件。該程序沒有任何輸出,但是你可以通過代碼研讀來理解程序的作用。

注意: 當序列化一個對象到文件時, 按照 Java 的標準約定是給文件一個 .ser 擴展名。

SerializeDemo.java 文件代碼:

importjava.io.*;publicclassSerializeDemo{publicstaticvoidmain(String[]args){Employeee =newEmployee();e.name = "Reyan Ali";e.address = "Phokka Kuan, Ambehta Peer";e.SSN = 11122333;e.number = 101;try{FileOutputStreamfileOut =newFileOutputStream("/tmp/employee.ser");ObjectOutputStreamout =newObjectOutputStream(fileOut);out.writeObject(e);out.close();fileOut.close();System.out.printf("Serialized data is saved in /tmp/employee.ser");}catch(IOExceptioni){i.printStackTrace();}}}

反序列化對象

下面的 DeserializeDemo 程序實例了反序列化,/tmp/employee.ser 存儲了 Employee 對象。

DeserializeDemo.java 文件代碼:

importjava.io.*;publicclassDeserializeDemo{publicstaticvoidmain(String[]args){Employeee =null;try{FileInputStreamfileIn =newFileInputStream("/tmp/employee.ser");ObjectInputStreamin =newObjectInputStream(fileIn);e =(Employee)in.readObject();in.close();fileIn.close();}catch(IOExceptioni){i.printStackTrace();return;}catch(ClassNotFoundExceptionc){System.out.println("Employee class not found");c.printStackTrace();return;}System.out.println("Deserialized Employee...");System.out.println("Name:" +e.name);System.out.println("Address:" +e.address);System.out.println("SSN:" +e.SSN);System.out.println("Number:" +e.number);}}

以上程序編譯運行結果如下所示:

Deserialized Employee...
Name: Reyan Ali
Address:Phokka Kuan, Ambehta Peer
SSN: 0
Number:101

這裏要注意以下要點:

readObject() 方法中的 try/catch代碼塊嘗試捕獲 ClassNotFoundException 異常。對於 JVM 可以反序列化對象,它必須是能夠找到字節碼的類。如果JVM在反序列化對象的過程中找不到該類,則拋出一個 ClassNotFoundException 異常。

注意,readObject() 方法的返回值被轉化成 Employee 引用。

當對象被序列化時,屬性 SSN 的值爲 111222333,但是因爲該屬性是短暫的,該值沒有被髮送到輸出流。所以反序列化後 Employee 對象的 SSN 屬性爲 0。

發佈了13 篇原創文章 · 獲贊 72 · 訪問量 11萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章