什么是序列化
(1)序列化是將對象轉(zhuǎn)變?yōu)樽止?jié)序列的過程,反序列化則是將字節(jié)序列恢復(fù)為對象的過程啄栓。
(2)對象序列化保存的是對象的狀態(tài),即它的成員變量妖混;
(3)對象的持久化存儲(寫文件)瘦材,網(wǎng)絡(luò)傳輸對象,或者使用RMI都會用到對象序列化椿猎。
JAVA 提供的操作序列化的接口
(1)Java 主要提供給了兩個接口實現(xiàn)對象的序列化和反序列化,java.io.ObjectInputStream的readObject()方法?和 java.io.ObjectOutputStream 的writeObject(Object obj)方法;
(2)只有實現(xiàn)Serializable或Externalizable接口的類的對象才能被序列化寿弱;否則會拋出java.io.NotSerializableException異常犯眠。
JAVA對象序列化示例
(1)類實現(xiàn) Serializable接口
?類中未定義 writeObject(Object obj)和readObject方法,那么按照默認的序列化方式實現(xiàn)序列化和反序列化症革。
以上代碼展示了如何序列化對象到一個文件中并從文件中反序列化的過程筐咧。
序列化的過程:
首先創(chuàng)建 ObjectOutputStream 對象,該對象可以包裝其他輸出流噪矛,比如文件輸出流量蕊;
調(diào)用對象輸出流的writeObject(Object obj)方法,可以將對象寫入到輸出流中艇挨。
關(guān)閉流残炮。結(jié)束。
對象持久化到文件中的過程結(jié)束缩滨。
反序列化的過程:
首先創(chuàng)建ObjectInputStream對象势就,類似于ObjectOutputStream;
調(diào)用對象輸入流的readObject()方法脉漏,讀對象到輸入流中苞冯。返回字節(jié)序列轉(zhuǎn)化的對象。
關(guān)閉流鸠删;結(jié)束抱完。
輸出:
?類中定義了 writeObject(Object obj)和readObject方法,那么按照自定義的序列化方實現(xiàn)式序列化和反序列化刃泡。
在Student.java添加如下兩個方法:
輸出結(jié)果:
(2)transient?關(guān)鍵字
? 當某個成員變量聲明為transient后巧娱,默認的序列化機制就會忽略該變量碉怔。
將age字段聲明為transient,
輸出 age=0:
此時我們可以選擇單獨傳輸某個字段禁添;修改writeObject和readObject方法:
結(jié)果:
單獨傳輸了age 字段撮胧,因此age=18;
補充:除了上面提到的兩個方法外:
private void writeObject(java.io.ObjectOutputStream out)?throws IOException ;
private void readObject(java.io.ObjectInputStream in)?throws IOException, ClassNotFoundException;
還有其他三個方法老翘,可供我們定制自己的序列化反序列化過程:
private void readObjectNoData()?throws ObjectStreamException;
ANY-ACCESS-MODIFIER Object writeReplace() throws ObjectStreamException;
ANY-ACCESS-MODIFIER Object readResolve() throws ObjectStreamException;
readObjectNoData() :用于初始化反序列化對象芹啥,當發(fā)生一些情況導(dǎo)致反序列化對象不能獲得數(shù)據(jù)時調(diào)用;
writeReplace() :指派其他對象寫入序列化的流中铺峭;
readResolve():返回的對象替換反序列化創(chuàng)建的實例墓怀;
readResolve() 常用于單例模式中;示例:
修改Student.java,添加instanceHoder:
修改SimpleSerial.java:
結(jié)果輸出:
可以看到卫键,s==student返回false傀履,也就是說反序列化后得到的Student對象并不是唯一的instance,因此這樣寫單例模式是失敗的莉炉;
修正:
再次運行:
總結(jié):
當進行序列化的時候:
首先JVM會先調(diào)用writeReplace方法,在這個階段,我們可以進行張冠李戴,將需要進行序列化的對象換成我們指定的對象.
跟著JVM將調(diào)用writeObject方法,來將對象中的屬性一個個進行序列化,我們可以在這個方法中控制住哪些屬性需要序列化.
當反序列化的時候:
JVM會調(diào)用readObject方法,將我們剛剛在writeObject方法序列化好的屬性,反序列化回來.
然后在readResolve方法中,我們也可以指定JVM返回我們特定的對象(不是剛剛序列化回來的對象).
注意到在writeReplace和readResolve,我們可以嚴格控制singleton的對象,在同一個JVM中完完全全只有唯一的對象,控制不讓singleton對象產(chǎn)生副本.
(3)類實現(xiàn)Externalizable 接口?
? ?Externalizable 接口繼承 Serializable接口:?
public interface Externalizable extends?Serializable?;
Serializable接口是一個mark interface钓账,沒有實際方法;而Externalizable 接口提供了兩個方法:
void ?readExternal?(ObjectInput?in) 絮宁;
void ?writeExternal?(ObjectOutput?out) ;
readExternal (ObjectInput in):從輸入流中讀取內(nèi)容恢復(fù)對象梆暮;
writeExternal?(ObjectOutput?out) : 寫入對象到輸出流中;
示例:
使用Externalizable進行序列化時绍昂,當讀取對象時啦粹,會調(diào)用被序列化類的無參構(gòu)造器去創(chuàng)建一個新的對象,然后再將被保存對象的字段的值分別填充到新對象中治专。因此卖陵,必須提供一個無參構(gòu)造器遭顶,訪問權(quán)限為public张峰;否則會拋出java.io.InvalidClassException 異常;
總結(jié):Externalizable接口實現(xiàn)的功能與Serializable接口類似棒旗,Serializable序列化時不會調(diào)用默認的構(gòu)造器喘批,而Externalizable序列化時會調(diào)用默認構(gòu)造器;