Java IO之序列化與反序列化詳解
1、什么是序列化與反序列化?
序列化:指把堆內(nèi)存中的 Java 對象數(shù)據(jù),通過某種方式把對象存儲到磁盤文件中或者傳遞給其他網(wǎng)絡(luò)節(jié)點(diǎn)(在網(wǎng)絡(luò)上傳輸)。這個(gè)過程稱為序列化。通俗來說就是將數(shù)據(jù)結(jié)構(gòu)或?qū)ο筠D(zhuǎn)換成二進(jìn)制串的過程
反序列化:把磁盤文件中的對象數(shù)據(jù)或者把網(wǎng)絡(luò)節(jié)點(diǎn)上的對象數(shù)據(jù),恢復(fù)成Java對象模型的過程。也就是將在序列化過程中所生成的二進(jìn)制串轉(zhuǎn)換成數(shù)據(jù)結(jié)構(gòu)或者對象的過程
2、為什么要做序列化?
①、在分布式系統(tǒng)中,此時(shí)需要把對象在網(wǎng)絡(luò)上傳輸,就得把對象數(shù)據(jù)轉(zhuǎn)換為二進(jìn)制形式,需要共享的數(shù)據(jù)的 JavaBean 對象,都得做序列化。
②、服務(wù)器鈍化:如果服務(wù)器發(fā)現(xiàn)某些對象好久沒活動了,那么服務(wù)器就會把這些內(nèi)存中的對象持久化在本地磁盤文件中(Java對象轉(zhuǎn)換為二進(jìn)制文件);如果服務(wù)器發(fā)現(xiàn)某些對象需要活動時(shí),先去內(nèi)存中尋找,找不到再去磁盤文件中反序列化我們的對象數(shù)據(jù),恢復(fù)成 Java 對象。這樣能節(jié)省服務(wù)器內(nèi)存。
3、Java 怎么進(jìn)行序列化?
①、需要做序列化的對象的類,必須實(shí)現(xiàn)序列化接口:Java.lang.Serializable 接口(這是一個(gè)標(biāo)志接口,沒有任何抽象方法),Java 中大多數(shù)類都實(shí)現(xiàn)了該接口,比如:String,Integer
②、底層會判斷,如果當(dāng)前對象是 Serializable 的實(shí)例,才允許做序列化,Java對象 instanceof Serializable 來判斷。
③、在 Java 中使用對象流來完成序列化和反序列化
ObjectOutputStream
:通過 writeObject()方法做序列化操作
ObjectInputStream:
通過 readObject() 方法做反序列化操作
第一步:創(chuàng)建一個(gè) JavaBean 對象
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 對象實(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();
我們打開 a.txt 文件,發(fā)現(xiàn)里面的內(nèi)容亂碼,注意這不需要我們來看懂,這是二進(jìn)制文件,計(jì)算機(jī)能讀懂就行了。
錯(cuò)誤一:如果新建的 Person 對象沒有實(shí)現(xiàn)Serializable 接口,那么上面的操作會報(bào)錯(cuò):
第三步:使用ObjectInputStream 對象實(shí)現(xiàn)反序列化
反序列化的對象必須要提供該對象的字節(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();
問題1:如果某些數(shù)據(jù)不需要做序列化,比如密碼,比如上面的年齡?
解決辦法:在字段面前加上transient
private String name;//需要序列化 transient private int age;//不需要序列化
那么我們在反序列化的時(shí)候,打印出來的就是Person [name=vae, age=0],整型數(shù)據(jù)默認(rèn)值為 0
問題2:序列化版本問題,在完成序列化操作后,由于項(xiàng)目的升級或修改,可能我們會對序列化對象進(jìn)行修改,比如增加某個(gè)字段,那么我們在進(jìn)行反序列化就會報(bào)錯(cuò):
解決辦法:在 JavaBean 對象中增加一個(gè) serialVersionUID 字段,用來固定這個(gè)版本,無論我們怎么修改,版本都是一致的,就能進(jìn)行反序列化了
private static final long serialVersionUID = 8656128222714547171L;
總結(jié)
本篇文章就到這里了,希望能夠給你帶來幫助,也希望您能夠多多關(guān)注腳本之家的更多內(nèi)容!
相關(guān)文章
Java中時(shí)間戳的獲取和轉(zhuǎn)換的示例分析
這篇文章主要介紹了Java中時(shí)間戳的獲取和轉(zhuǎn)換的示例分析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-07-07Java中使用BeanMap將對象轉(zhuǎn)為Map詳解
這篇文章主要介紹了Java中使用BeanMap將對象轉(zhuǎn)為Map詳解,BeanMap?是?Apache?Commons?BeanUtils?庫中的一個(gè)類,BeanMap?可以將?Java?對象的屬性作為鍵,屬性值作為對應(yīng)的值,存儲在一個(gè)?Map?中,它提供了一種將?Java?對象轉(zhuǎn)換為?Map?的方式,需要的朋友可以參考下2024-01-01Springcloud GateWay網(wǎng)關(guān)配置過程圖解
這篇文章主要介紹了Springcloud GateWay網(wǎng)關(guān)配置過程圖解,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-12-12Java實(shí)現(xiàn)去掉字符串重復(fù)字母的方法示例
這篇文章主要介紹了Java實(shí)現(xiàn)去掉字符串重復(fù)字母的方法,涉及java針對字符串的遍歷、判斷、運(yùn)算等相關(guān)操作技巧,需要的朋友可以參考下2017-12-12