Java 序列化
Java 提供了一種對(duì)象序列化的機(jī)制转培,該機(jī)制中菜循,一個(gè)對(duì)象可以被表示為一個(gè)字節(jié)序列倾贰,該字節(jié)序列包括該對(duì)象的數(shù)據(jù)束析、有關(guān)對(duì)象的類型的信息和存儲(chǔ)在對(duì)象中數(shù)據(jù)的類型艳馒。
將序列化對(duì)象寫入文件之后,可以從文件中讀取出來(lái)员寇,并且對(duì)它進(jìn)行反序列化弄慰,也就是說(shuō),對(duì)象的類型信息蝶锋、對(duì)象的數(shù)據(jù)陆爽,還有對(duì)象中的數(shù)據(jù)類型可以用來(lái)在內(nèi)存中新建對(duì)象。
整個(gè)過(guò)程都是 Java 虛擬機(jī)(JVM)獨(dú)立的扳缕,也就是說(shuō)慌闭,在一個(gè)平臺(tái)上序列化的對(duì)象可以在另一個(gè)完全不同的平臺(tái)上反序列化該對(duì)象。
類 ObjectInputStream 和 ObjectOutputStream 是高層次的數(shù)據(jù)流躯舔,它們包含序列化和反序列化對(duì)象的方法驴剔。
ObjectOutputStream 類包含很多寫方法來(lái)寫各種數(shù)據(jù)類型,但是一個(gè)特別的方法例外:
public final void writeObject(Object x) throws IOException
上面的方法序列化一個(gè)對(duì)象粥庄,并將它發(fā)送到輸出流丧失。相似的 ObjectInputStream 類包含如下反序列化一個(gè)對(duì)象的方法:
public final Object readObject() throws IOException,
ClassNotFoundException
該方法從流中取出下一個(gè)對(duì)象,并將對(duì)象反序列化惜互。它的返回值為Object布讹,因此,你需要將它轉(zhuǎn)換成合適的數(shù)據(jù)類型训堆。
為了演示序列化在Java中是怎樣工作的描验,我將使用之前教程中提到的Employee類,假設(shè)我們定義了如下的Employee類坑鱼,該類實(shí)現(xiàn)了Serializable 接口膘流。
Employee.java 文件代碼:
public class Employee implements java.io.Serializable
{
public String name;
public String address;
public transient int SSN;
public int number;
public void mailCheck()
{
System.out.println("Mailing a check to " + name
+ " " + address);
}
}
請(qǐng)注意,一個(gè)類的對(duì)象要想序列化成功,必須滿足兩個(gè)條件:
該類必須實(shí)現(xiàn) java.io.Serializable 對(duì)象睡扬。
該類的所有屬性必須是可序列化的盟蚣。如果有一個(gè)屬性不是可序列化的黍析,則該屬性必須注明是短暫的卖怜。
如果你想知道一個(gè) Java 標(biāo)準(zhǔn)類是否是可序列化的,請(qǐng)查看該類的文檔阐枣。檢驗(yàn)一個(gè)類的實(shí)例是否能序列化十分簡(jiǎn)單马靠, 只需要查看該類有沒有實(shí)現(xiàn) java.io.Serializable接口。
序列化對(duì)象
ObjectOutputStream 類用來(lái)序列化一個(gè)對(duì)象蔼两,如下的 SerializeDemo 例子實(shí)例化了一個(gè) Employee 對(duì)象甩鳄,并將該對(duì)象序列化到一個(gè)文件中
該程序執(zhí)行后,就創(chuàng)建了一個(gè)名為 employee.ser 文件
該程序沒有任何輸出额划,我們可以通過(guò)代碼研讀來(lái)理解程序的作用
當(dāng)序列化一個(gè)對(duì)象到文件時(shí)妙啃, 會(huì)按照 Java 的標(biāo)準(zhǔn)約定是給文件一個(gè) .ser 擴(kuò)展名
SerializeDemo.java
import java.io.*;
public class SerializeDemo
{
public static void main(String [] args)
{
Employee e = new Employee();
e.name = "Reyan Ali";
e.address = "Phokka Kuan, Ambehta Peer";
e.SSN = 11122333;
e.number = 101;
try
{
FileOutputStream fileOut =
new FileOutputStream("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(IOException i)
{
i.printStackTrace();
}
}
}
反序列化對(duì)象
現(xiàn)在我們來(lái)實(shí)現(xiàn)一個(gè) DeserializeDemo 類用于反序列化
文件 employee.ser
存儲(chǔ)了 Employee 對(duì)象
DeserializeDemo.java
import java.io.*;
public class DeserializeDemo
{
public static void main(String [] args)
{
Employee e = null;
try
{
FileInputStream fileIn = new FileInputStream("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);
}
}
編譯運(yùn)行以上 Java 代碼,輸出結(jié)果如下
Deserialized Employee...
Name: Reyan Ali
Address:Phokka Kuan, Ambehta Peer
SSN: 0
Number:101
在這段代碼中俊戳,有幾點(diǎn)需要注意
readObject() 方法中的 try/catch 代碼塊嘗試捕獲 ClassNotFoundException 異常
對(duì)于 JVM 可以反序列化對(duì)象揖赴,它必須是能夠找到字節(jié)碼的類
如果 JVM 在反序列化對(duì)象的過(guò)程中找不到該類,則拋出一個(gè) ClassNotFoundException 異常
readObject() 方法的返回值被轉(zhuǎn)化成 Employee 引用
當(dāng)對(duì)象被序列化時(shí)抑胎,屬性 SSN 的值為 111222333燥滑,但是因?yàn)樵搶傩允嵌虝旱模撝禌]有被發(fā)送到輸出流
所以反序列化后 Employee 對(duì)象的 SSN 屬性為 0