Java 提供了一種對象序列化的機制,該機制中,一個對象可以被表示為一個字節(jié)序列街氢,該字節(jié)序列包括該對象的數(shù)據(jù)、有關對象的類型的信息和存儲在對象中數(shù)據(jù)的類型睦袖。
將序列化對象寫入文件之后,可以從文件中讀取出來荣刑,并且對它進行反序列化馅笙,也就是說伦乔,對象的類型信息、對象的數(shù)據(jù)董习,還有對象中的數(shù)據(jù)類型可以用來在內(nèi)存中新建對象烈和。
整個過程都是 Java 虛擬機(JVM)獨立的,也就是說皿淋,在一個平臺上序列化的對象可以在另一個完全不同的平臺上反序列化該對象招刹。
類 ObjectInputStream 和 ObjectOutputStream 是高層次的數(shù)據(jù)流,它們包含反序列化和序列化對象的方法窝趣。
ObjectOutputStream 類包含很多寫方法來寫各種數(shù)據(jù)類型疯暑,但是一個特別的方法例外:
public final void writeObject(Object x) throws IOException
上面的方法序列化一個對象,并將它發(fā)送到輸出流哑舒。相似的 ObjectInputStream 類包含如下反序列化一個對象的方法:
public final Object readObject() throws IOException,
ClassNotFoundException
該方法從流中取出下一個對象妇拯,并將對象反序列化。它的返回值為Object洗鸵,因此越锈,你需要將它轉(zhuǎn)換成合適的數(shù)據(jù)類型。
為了演示序列化在Java中是怎樣工作的膘滨,我將使用之前教程中提到的Employee類甘凭,假設我們定義了如下的Employee類,該類實現(xiàn)了Serializable 接口火邓。
請注意丹弱,一個類的對象要想序列化成功,必須滿足兩個條件:
該類必須實現(xiàn) java.io.Serializable
對象贡翘。
該類的所有屬性必須是可序列化的蹈矮。如果有一個屬性不是可序列化的,則該屬性必須注明是短暫的泛鸟。
如果你想知道一個 Java 標準類是否是可序列化的,請查看該類的文檔踊东。檢驗一個類的實例是否能序列化十分簡單北滥, 只需要查看該類有沒有實現(xiàn) java.io.Serializable
接口。
序列化對象
ObjectOutputStream 類用來序列化一個對象闸翅,如下的 SerializeDemo 例子實例化了一個 Employee 對象再芋,并將該對象序列化到一個文件中。
該程序執(zhí)行后坚冀,就創(chuàng)建了一個名為 employee.ser 文件济赎。該程序沒有任何輸出,但是你可以通過代碼研讀來理解程序的作用。
注意: 當序列化一個對象到文件時司训, 按照 Java 的標準約定是給文件一個 .ser 擴展名构捡。
import java.io.*;
// 序列化對象
public class SerializeDemo {
public static void main(String[] args) {
Employee e = new Employee();
e.name = "shavekevin";
e.address = "北京 朝陽";
e.SSN = 123;
e.number = 9527;
try {
FileOutputStream fileOut =
new FileOutputStream("/tmp/employee.ser");
ObjectOutputStream out = new ObjectOutputStream(fileOut);
out.writeObject(e);
out.close();
fileOut.close();
System.out.printf("Serialized data is saved in /tmp/employee.ser");
} catch (Exception i) {
// TODO: handle exception
}
}
}
編譯后的結(jié)果為:
Serialized data is saved in /tmp/employee.ser
反序列化對象
下面的 DeserializeDemo 程序?qū)嵗朔葱蛄谢?tmp/employee.ser 存儲了 Employee 對象。
import java.io.*;
// 反序列化對象
public class DeserializeDemo {
public static void main(String[] args) {
Employee e = null;
try {
FileInputStream fileIn = new FileInputStream("/tmp/employee.ser");
ObjectInputStream in = new ObjectInputStream(fileIn);
e = (Employee) in.readObject();
in.close();
fileIn.close();
} catch (IOException i) {
i.printStackTrace();
return;
} catch (ClassNotFoundException c) {
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);
}
}
編譯后的結(jié)果為:
Deserialized Employee...
Name: shavekevin
Address: 北京 朝陽
SSN: 0
Number: 9527
這里要注意以下要點:
這里要注意以下要點:
readObject() 方法中的 try/catch代碼塊嘗試捕獲 ClassNotFoundException 異常壳猜。對于 JVM 可以反序列化對象勾徽,它必須是能夠找到字節(jié)碼的類。如果JVM在反序列化對象的過程中找不到該類统扳,則拋出一個 ClassNotFoundException 異常喘帚。
注意,readObject() 方法的返回值被轉(zhuǎn)化成 Employee 引用咒钟。
當對象被序列化時吹由,屬性 SSN 的值為 123,但是因為該屬性是短暫的盯腌,該值沒有被發(fā)送到輸出流溉知。所以反序列化后 Employee 對象的 SSN 屬性為 0。