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 write Object(Objectx) throws IOException
上面的方法序列化一個(gè)對(duì)象,并將它發(fā)送到輸出流。相似的 ObjectInputStream 類包含如下反序列化一個(gè)對(duì)象的方法:
public final Object read Object() throws IOException, ClassNotFoundException
該方法從流中取出下一個(gè)對(duì)象,并將對(duì)象反序列化。它的返回值為Object,因此,你需要將它轉(zhuǎn)換成合適的數(shù)據(jù)類型。
為了演示序列化在Java中是怎樣工作的,假設(shè)我們定義了如下的Employee類,該類實(shí)現(xiàn)了Serializable 接口。
<pre><code>
public class Employee implements Serializable {
public String name;
public int age;
public transient int SS;
public String address;
public void printInfo() {
System.out.println("information is : " + name + " " + age + SS);
}
}
</code></pre>
對(duì)該Employee類進(jìn)行序列化
<pre><code>
Employee e = new Employee();
e.name = "xiaoming";
e.address = "beijing";
e.age = 23;
e.SS = 11;
try {
FileOutputStream outputStream = new FileOutputStream("/tmp/employee.ser");
try {
ObjectOutputStream objectOutputStream = new ObjectOutputStream(outputStream);
objectOutputStream.writeObject(e);
outputStream.close();
objectOutputStream.close();
System.out.printf("Serialized data is saved in /tmp/employee.ser");
} catch (IOException e1) {
e1.printStackTrace();
}
} catch (FileNotFoundException e1) {
e1.printStackTrace();
}
</code></pre>
對(duì)Employee類進(jìn)行反序列化
<pre><code>
Employee e= null;
try {
FileInputStream inputStream = new FileInputStream("/tmp/employee.ser");
try {
ObjectInputStream objectInputStream = new ObjectInputStream(inputStream);
try {
e = (Employee) objectInputStream.readObject();
objectInputStream.close();
inputStream.close();
} catch (ClassNotFoundException e1) {
System.out.println("Employee class not found");
e1.printStackTrace();
return;
}
} catch (IOException e1) {
e1.printStackTrace();
}
} catch (FileNotFoundException e1) {
System.out.println("Employee class not found");
e1.printStackTrace();
return;
}
System.out.println("Deserialized Employee...");
System.out.println("Name: " + e.name);
System.out.println("Address: " + e.address);
System.out.println("SS: " + e.SS);
System.out.println("Age: " + e.age);
</code></pre>
反序列化打印結(jié)果

通過(guò)結(jié)果,可以發(fā)現(xiàn)定義序列化的字段都返回之前設(shè)定的值,而定義不需要序列化的字段SS返回了0,而不是之前設(shè)定的11。
請(qǐng)注意
一個(gè)類的對(duì)象要想序列化成功,必須滿足兩個(gè)條件:
1、該類必須實(shí)現(xiàn) java.io.Serializable 對(duì)象。
2、該類的所有屬性必須是可序列化的。如果有一個(gè)屬性不是可序列化的,則該屬性必須注明是短暫的。如果你想知道一個(gè) Java 標(biāo)準(zhǔn)類是否是可序列化的,請(qǐng)查看該類的文檔。檢驗(yàn)一個(gè)類的實(shí)例是否能序列化十分簡(jiǎn)單, 只需要查看該類有沒(méi)有實(shí)現(xiàn) java.io.Serializable接口。
3、readObject() 方法中的 try/catch代碼塊嘗試捕獲 ClassNotFoundException 異常。對(duì)于 JVM 可以反序列化對(duì)象,它必須是能夠找到字節(jié)碼的類。如果JVM在反序列化對(duì)象的過(guò)程中找不到該類,則拋出一個(gè) ClassNotFoundException 異常。
注意,readObject() 方法的返回值被轉(zhuǎn)化成 Employee 引用。
當(dāng)對(duì)象被序列化時(shí),屬性 SS 的值為 11,但是因?yàn)樵搶傩允嵌虝旱模撝禌](méi)有被發(fā)送到輸出流。所以反序列化后 Employee 對(duì)象的 SSN 屬性為 0。
本文僅為學(xué)習(xí)筆記,文章整理來(lái)源
http://www.runoob.com/java/java-multithreading.html