詳解Java對象序列化為什么要使用SerialversionUID
1、首先談?wù)劄槭裁匆蛄谢瘜ο?/strong>
- 把對象轉(zhuǎn)換為字節(jié)序列的過程稱為對象的序列化。
- 把字節(jié)序列恢復(fù)為對象的過程稱為對象的反序列化。
對象的序列化主要有兩種用途:
1) 把對象的字節(jié)序列永久地保存到硬盤上,通常存放在一個(gè)文件中;
2) 在網(wǎng)絡(luò)上傳送對象的字節(jié)序列。
在很多應(yīng)用中,需要對某些對象進(jìn)行序列化,讓它們離開內(nèi)存空間,入住物理硬盤,以便長期保存。比如最常見的是Web服務(wù)器中的Session對象,當(dāng)有 10萬用戶并發(fā)訪問,就有可能出現(xiàn)10萬個(gè)Session對象,內(nèi)存可能吃不消,于是Web容器就會(huì)把一些seesion先序列化到硬盤中,等要用了,再把保存在硬盤中的對象還原到內(nèi)存中。
當(dāng)兩個(gè)進(jìn)程在進(jìn)行遠(yuǎn)程通信時(shí),彼此可以發(fā)送各種類型的數(shù)據(jù)。無論是何種類型的數(shù)據(jù),都會(huì)以二進(jìn)制序列的形式在網(wǎng)絡(luò)上傳送。發(fā)送方需要把這個(gè)Java對象轉(zhuǎn)換為字節(jié)序列,才能在網(wǎng)絡(luò)上傳送;接收方則需要把字節(jié)序列再恢復(fù)為Java對象
2、為什么要使用SerialversionUID呢
簡單看一下 Serializable接口的說明
If a serializable class does not explicitly declare a serialVersionUID,
then the serialization runtime will calculate a default
serialVersionUID value for that class based on various aspects of the class,
as described in the Java(TM) Object Serialization Specification.
如果用戶沒有自己聲明一個(gè)serialVersionUID,接口會(huì)默認(rèn)生成一個(gè)serialVersionUID
However, it is stronglyrecommended that all serializable classes explicitly declareserialVersionUID values, since the default serialVersionUID computation is highly sensitive to class details that may vary depending on compiler implementations, and can thus result in unexpectedInvalidClassException
s during deserialization.
但是強(qiáng)烈建議用戶自定義一個(gè)serialVersionUID,因?yàn)槟J(rèn)的serialVersinUID對于class的細(xì)節(jié)非常敏感,反序列化時(shí)可能會(huì)導(dǎo)致InvalidClassException這個(gè)異常。
e.g:1.使用默認(rèn)的serialVersionUID
我們先建一個(gè)實(shí)體類Person 實(shí)現(xiàn)Serializable接口
public class Person implements Serializable { private int age; private String name; private String sex; public int getAge() { return age; } public void setAge(int age) { this.age = age; } public String getName() { return name; } public void setName(String name) { this.name = name; } public String getSex() { return sex; } public void setSex(String sex) { this.sex = sex; } }
然后去序列化和反序列化它
public class Person implements Serializable { private int age; private String name; private String sex; public int getAge() { return age; } public void setAge(int age) { this.age = age; } public String getName() { return name; } public void setName(String name) { this.name = name; } public String getSex() { return sex; } public void setSex(String sex) { this.sex = sex; } }
結(jié)果如圖
e.g:2
如果我們先盡心序列化,然后在反序列化之前修改了Person類會(huì)怎樣呢
public class Person implements Serializable { private int age; private String name; private String sex; private String address; public String getAddress() { return address; } public void setAddress(String address) { this.address = address; } public int getAge() { return age; } public void setAge(int age) { this.age = age; } public String getName() { return name; } public void setName(String name) { this.name = name; } public String getSex() { return sex; } public void setSex(String sex) { this.sex = sex; } }
運(yùn)行結(jié)果
可以看到,當(dāng)我們修改Person類的時(shí)候,Person類對應(yīng)的SerialversionUID也變化了,而序列化和反序列化就是通過對比其SerialversionUID來進(jìn)行的,一旦SerialversionUID不匹配,反序列化就無法成功。在實(shí)際的生產(chǎn)環(huán)境中,我們可能會(huì)建一系列的中間Object來反序列化我們的pojo,為了解決這個(gè)問題,我們就需要在實(shí)體類中自定義SerialversionUID。
e.g:3 在Person類中加入自定義SerialversionUID
public class Person implements Serializable { private static final long serialVersionUID = -5809782578272943999L; private int age; private String name; private String sex; private String address; public String getAddress() { return address; } public void setAddress(String address) { this.address = address; } public int getAge() { return age; } public void setAge(int age) { this.age = age; } public String getName() { return name; } public void setName(String name) { this.name = name; } public String getSex() { return sex; } public void setSex(String sex) { this.sex = sex; } }
不管我們序列化之后如何更改我們的Person(不刪除原有字段),最終都可以反序列化成功。
到此這篇關(guān)于詳解Java對象序列化為什么要使用SerialversionUID的文章就介紹到這了,更多相關(guān)Java對象序列化SerialversionUID內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java定時(shí)器通信協(xié)議管理模塊Timer詳解
這篇文章主要介紹了Java定時(shí)器通信協(xié)議管理模塊Timer,?Timer一般指定時(shí)器(通信協(xié)議管理模塊)人類最早使用的定時(shí)工具是沙漏或水漏,但在鐘表誕生發(fā)展成熟之后,人們開始嘗試使用這種全新的計(jì)時(shí)工具來改進(jìn)定時(shí)器,達(dá)到準(zhǔn)確控制時(shí)間的目的2022-08-08Java整合Jackson實(shí)現(xiàn)反序列化器流程
Jackson是一個(gè)開源的Java序列化和反序列化工具,可以將Java對象序列化為XML或JSON格式的字符串,以及將XML或JSON格式的字符串反序列化為Java對象。由于其使用簡單,速度較快,且不依靠除JDK外的其他庫,被眾多用戶所使用2023-01-01java處理數(shù)據(jù)庫不支持的emoji表情符問題解決
這篇文章主要介紹了java處理數(shù)據(jù)庫不支持的emoji表情符問題解決,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-09-09Springboot ApplicationRunner的使用解讀
這篇文章主要介紹了Springboot ApplicationRunner的使用解讀,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-05-05SkyWalking自定義鏈路追蹤實(shí)現(xiàn)步驟
本文詳細(xì)介紹了如何使用SkyWalking進(jìn)行自定義鏈路追蹤的步驟,包括POM文件依賴和邏輯業(yè)務(wù)代碼的編寫,文章最后推薦了腳本之家作為進(jìn)一步學(xué)習(xí)的資源2024-02-02