Java IO之序列化與反序列化詳解
1、什么是序列化與反序列化?
序列化:指把堆內(nèi)存中的 Java 對(duì)象數(shù)據(jù),通過(guò)某種方式把對(duì)象存儲(chǔ)到磁盤文件中或者傳遞給其他網(wǎng)絡(luò)節(jié)點(diǎn)(在網(wǎng)絡(luò)上傳輸)。這個(gè)過(guò)程稱為序列化。通俗來(lái)說(shuō)就是將數(shù)據(jù)結(jié)構(gòu)或?qū)ο筠D(zhuǎn)換成二進(jìn)制串的過(guò)程
反序列化:把磁盤文件中的對(duì)象數(shù)據(jù)或者把網(wǎng)絡(luò)節(jié)點(diǎn)上的對(duì)象數(shù)據(jù),恢復(fù)成Java對(duì)象模型的過(guò)程。也就是將在序列化過(guò)程中所生成的二進(jìn)制串轉(zhuǎn)換成數(shù)據(jù)結(jié)構(gòu)或者對(duì)象的過(guò)程
2、為什么要做序列化?
①、在分布式系統(tǒng)中,此時(shí)需要把對(duì)象在網(wǎng)絡(luò)上傳輸,就得把對(duì)象數(shù)據(jù)轉(zhuǎn)換為二進(jìn)制形式,需要共享的數(shù)據(jù)的 JavaBean 對(duì)象,都得做序列化。
②、服務(wù)器鈍化:如果服務(wù)器發(fā)現(xiàn)某些對(duì)象好久沒(méi)活動(dòng)了,那么服務(wù)器就會(huì)把這些內(nèi)存中的對(duì)象持久化在本地磁盤文件中(Java對(duì)象轉(zhuǎn)換為二進(jìn)制文件);如果服務(wù)器發(fā)現(xiàn)某些對(duì)象需要活動(dòng)時(shí),先去內(nèi)存中尋找,找不到再去磁盤文件中反序列化我們的對(duì)象數(shù)據(jù),恢復(fù)成 Java 對(duì)象。這樣能節(jié)省服務(wù)器內(nèi)存。
3、Java 怎么進(jìn)行序列化?
①、需要做序列化的對(duì)象的類,必須實(shí)現(xiàn)序列化接口:Java.lang.Serializable 接口(這是一個(gè)標(biāo)志接口,沒(méi)有任何抽象方法),Java 中大多數(shù)類都實(shí)現(xiàn)了該接口,比如:String,Integer
②、底層會(huì)判斷,如果當(dāng)前對(duì)象是 Serializable 的實(shí)例,才允許做序列化,Java對(duì)象 instanceof Serializable 來(lái)判斷。
③、在 Java 中使用對(duì)象流來(lái)完成序列化和反序列化
ObjectOutputStream:通過(guò) writeObject()方法做序列化操作
ObjectInputStream:通過(guò) readObject() 方法做反序列化操作

第一步:創(chuàng)建一個(gè) JavaBean 對(duì)象
public class Person implements Serializable{
private String name;
private int age;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
@Override
public String toString() {
return "Person [name=" + name + ", age=" + age + "]";
}
public Person(String name, int age) {
super();
this.name = name;
this.age = age;
}
}
第二步:使用 ObjectOutputStream 對(duì)象實(shí)現(xiàn)序列化
//在根目錄下新建一個(gè) io 的文件夾
OutputStream op = new FileOutputStream("io"+File.separator+"a.txt");
ObjectOutputStream ops = new ObjectOutputStream(op);
ops.writeObject(new Person("vae",1));
ops.close();我們打開(kāi) a.txt 文件,發(fā)現(xiàn)里面的內(nèi)容亂碼,注意這不需要我們來(lái)看懂,這是二進(jìn)制文件,計(jì)算機(jī)能讀懂就行了。
錯(cuò)誤一:如果新建的 Person 對(duì)象沒(méi)有實(shí)現(xiàn)Serializable 接口,那么上面的操作會(huì)報(bào)錯(cuò):

第三步:使用ObjectInputStream 對(duì)象實(shí)現(xiàn)反序列化
反序列化的對(duì)象必須要提供該對(duì)象的字節(jié)碼文件.class
InputStream in = new FileInputStream("io"+File.separator+"a.txt");
ObjectInputStream os = new ObjectInputStream(in);
byte[] buffer = new byte[10];
int len = -1;
Person p = (Person) os.readObject();
System.out.println(p); //Person [name=vae, age=1]
os.close();問(wèn)題1:如果某些數(shù)據(jù)不需要做序列化,比如密碼,比如上面的年齡?
解決辦法:在字段面前加上transient
private String name;//需要序列化
transient private int age;//不需要序列化那么我們?cè)诜葱蛄谢臅r(shí)候,打印出來(lái)的就是Person [name=vae, age=0],整型數(shù)據(jù)默認(rèn)值為 0
問(wèn)題2:序列化版本問(wèn)題,在完成序列化操作后,由于項(xiàng)目的升級(jí)或修改,可能我們會(huì)對(duì)序列化對(duì)象進(jìn)行修改,比如增加某個(gè)字段,那么我們?cè)谶M(jìn)行反序列化就會(huì)報(bào)錯(cuò):


解決辦法:在 JavaBean 對(duì)象中增加一個(gè) serialVersionUID 字段,用來(lái)固定這個(gè)版本,無(wú)論我們?cè)趺葱薷?,版本都是一致的,就能進(jìn)行反序列化了
private static final long serialVersionUID = 8656128222714547171L;
總結(jié)
本篇文章就到這里了,希望能夠給你帶來(lái)幫助,也希望您能夠多多關(guān)注腳本之家的更多內(nèi)容!
相關(guān)文章
Java服務(wù)端性能優(yōu)化之JVM垃圾回收策略詳解
JVM垃圾回收策略涵蓋了基本原理、常見(jiàn)策略(如SerialGC、ParallelGC、CMS、G1GC)以及優(yōu)化建議,選擇合適的策略和調(diào)整參數(shù),如堆大小和GC日志,可以提高應(yīng)用性能和響應(yīng)速度,持續(xù)監(jiān)控和分析是關(guān)鍵步驟2025-03-03
完美解決java.lang.OutOfMemoryError處理錯(cuò)誤的問(wèn)題
下面小編就為大家?guī)?lái)一篇完美解決java.lang.OutOfMemoryError處理錯(cuò)誤的問(wèn)題。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-01-01
Spring?Security權(quán)限想要細(xì)化到按鈕實(shí)現(xiàn)示例
這篇文章主要為大家介紹了Spring?Security權(quán)限想要細(xì)化到按鈕實(shí)現(xiàn)示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-07-07
mybatis 獲取無(wú)數(shù)據(jù)的字段不顯示的問(wèn)題
這篇文章主要介紹了mybatis 獲取無(wú)數(shù)據(jù)的字段不顯示的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-07-07
java 中String和StringBuffer與StringBuilder的區(qū)別及使用方法
這篇文章主要介紹了java 中String和StringBuffer與StringBuilder的區(qū)別及使用方法的相關(guān)資料,在開(kāi)發(fā)過(guò)程中經(jīng)常會(huì)用到String 這個(gè)類進(jìn)行操作,需要的朋友可以參考下2017-08-08
Java中comparator接口和Comparable接口的比較解析
這篇文章主要介紹了Java中comparator接口和Comparable接口的比較解析,Java提供了一個(gè)用于比較的接口Comparator和Comparable接口,提供了一個(gè)比較的方法,所有實(shí)現(xiàn)該接口的類,都動(dòng)態(tài)的實(shí)現(xiàn)了該比較方法,需要的朋友可以參考下2023-08-08
Spring動(dòng)態(tài)數(shù)據(jù)源實(shí)現(xiàn)讀寫分離詳解
這篇文章主要為大家詳細(xì)介紹了Spring動(dòng)態(tài)數(shù)據(jù)源實(shí)現(xiàn)讀寫分離,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-07-07
關(guān)于Unsupported major.minor version 49.0的錯(cuò)誤解決辦法
這篇文章主要介紹了關(guān)于Unsupported major.minor version 49.0的錯(cuò)誤解決辦法的相關(guān)資料,需要的朋友可以參考下2015-11-11

