JDK自帶的序列化方式優(yōu)缺點(diǎn)及實(shí)現(xiàn)原理面試精講
1. 什么是 JDK 自帶的序列化方式?
JDK(Java Development Kit)提供了一種內(nèi)置的序列化機(jī)制,即 Java 對(duì)象序列化。它允許將 Java 對(duì)象轉(zhuǎn)換為字節(jié)流,以便在網(wǎng)絡(luò)上傳輸或保存到磁盤上,并且可以通過反序列化將字節(jié)流重新轉(zhuǎn)換回 Java 對(duì)象。
2. 為什么需要 JDK 自帶的序列化方式?
JDK 自帶的序列化方式具有以下幾個(gè)優(yōu)點(diǎn):
- 簡(jiǎn)單易用:使用 JDK 自帶的序列化方式,只需實(shí)現(xiàn) Serializable 接口并添加 serialVersionUID 字段即可。
- 跨平臺(tái):由于序列化后的數(shù)據(jù)是以字節(jié)流形式存儲(chǔ)的,因此可以在不同操作系統(tǒng)和編程語言之間進(jìn)行傳輸和共享。
- 對(duì)象圖形完整性:JDK 自帶的序列化方式會(huì)保留對(duì)象的完整結(jié)構(gòu),包括對(duì)象的屬性、方法等信息。
- 支持版本兼容性:當(dāng)類的定義發(fā)生變化時(shí),JDK 自帶的序列化方式能夠處理舊版本與新版本之間的兼容性問題。
3. JDK 自帶的序列化方式的實(shí)現(xiàn)原理?
JDK 自帶的序列化方式基于 Java 的反射機(jī)制和字節(jié)碼技術(shù)來實(shí)現(xiàn)。當(dāng)一個(gè)對(duì)象被序列化時(shí),JVM 會(huì)根據(jù)對(duì)象的類型和結(jié)構(gòu)將其轉(zhuǎn)換為字節(jié)流。在反序列化時(shí),JVM 會(huì)根據(jù)字節(jié)流中的信息重新構(gòu)造出原始對(duì)象。
具體實(shí)現(xiàn)過程如下:
- 序列化:當(dāng)調(diào)用
ObjectOutputStream
的writeObject()
方法時(shí),JVM 會(huì)檢查被序列化的對(duì)象是否實(shí)現(xiàn)了 Serializable 接口。如果沒有實(shí)現(xiàn),則拋出NotSerializableException
異常。如果實(shí)現(xiàn)了 Serializable 接口,則 JVM 會(huì)通過反射機(jī)制獲取對(duì)象的類結(jié)構(gòu),并將其轉(zhuǎn)換為字節(jié)流。 - 反序列化:當(dāng)調(diào)用
ObjectInputStream
的readObject()
方法時(shí),JVM 會(huì)讀取字節(jié)流并還原成 Java 對(duì)象。在此過程中,JVM 會(huì)使用反射機(jī)制創(chuàng)建對(duì)象,并通過反射設(shè)置對(duì)象的屬性值。
4. JDK 自帶的序列化方式的使用示例
以下是一個(gè)簡(jiǎn)單的示例代碼,演示了如何使用 JDK 自帶的序列化方式:
import java.io.*; public class SerializationExample { public static void main(String[] args) { // 創(chuàng)建一個(gè)Person對(duì)象 Person person = new Person("John", 25); try { // 將Person對(duì)象序列化到文件中 FileOutputStream fileOut = new FileOutputStream("person.ser"); ObjectOutputStream out = new ObjectOutputStream(fileOut); out.writeObject(person); out.close(); fileOut.close(); // 從文件中反序列化Person對(duì)象 FileInputStream fileIn = new FileInputStream("person.ser"); ObjectInputStream in = new ObjectInputStream(fileIn); Person deserializedPerson = (Person) in.readObject(); in.close(); fileIn.close(); // 打印反序列化后的Person對(duì)象 System.out.println("Deserialized Person:"); System.out.println("Name: " + deserializedPerson.getName()); System.out.println("Age: " + deserializedPerson.getAge()); } catch (IOException | ClassNotFoundException e) { e.printStackTrace(); } } } class Person implements Serializable { private static final long serialVersionUID = 1L; private String name; private int age; public Person(String name, int age) { this.name = name; this.age = age; } public String getName() { return name; } public int getAge() { return age; } }
5. JDK 自帶的序列化方式的優(yōu)點(diǎn)
- 簡(jiǎn)單易用:只需實(shí)現(xiàn) Serializable 接口即可進(jìn)行序列化和反序列化操作。
- 跨平臺(tái):可以在不同操作系統(tǒng)和編程語言之間傳輸和共享數(shù)據(jù)。
- 對(duì)象圖形完整性:保留了對(duì)象的完整結(jié)構(gòu),包括屬性、方法等信息。
- 版本兼容性:能夠處理舊版本與新版本之間的兼容性問題。
6. JDK 自帶的序列化方式的缺點(diǎn)
- 序列化后的字節(jié)流較大:JDK 自帶的序列化方式會(huì)將對(duì)象的所有屬性都轉(zhuǎn)換為字節(jié)流,導(dǎo)致序列化后的字節(jié)流較大,占用存儲(chǔ)空間較多。
- 可讀性差:序列化后的字節(jié)流是二進(jìn)制數(shù)據(jù),不易閱讀和理解。
- 性能較低:JDK 自帶的序列化方式在序列化和反序列化過程中需要進(jìn)行大量的 IO 操作和對(duì)象創(chuàng)建操作,性能相對(duì)較低。
7. JDK 自帶的序列化方式的使用注意事項(xiàng)
- 序列化 ID(serialVersionUID)的一致性:為了保證序列化后的字節(jié)流與反序列化時(shí)的類定義一致,需要顯式地指定一個(gè)序列化 ID。如果沒有指定,則 JVM 會(huì)根據(jù)類的結(jié)構(gòu)生成一個(gè)默認(rèn)的序列化 ID,但當(dāng)類的結(jié)構(gòu)發(fā)生變化時(shí),可能導(dǎo)致反序列化失敗。
- 非序列化字段的處理:如果一個(gè)類中包含非序列化字段,可以通過
transient
關(guān)鍵字將其標(biāo)記為瞬態(tài)字段,這樣在序列化過程中會(huì)被忽略。 - 版本兼容性:當(dāng)類的定義發(fā)生變化時(shí),需要考慮舊版本與新版本之間的兼容性問題??梢酝ㄟ^修改序列化 ID、添加
readObject()
和writeObject()
方法等方式來實(shí)現(xiàn)版本兼容性。
8. 總結(jié)
JDK 自帶的序列化方式提供了一種簡(jiǎn)單易用的 Java 對(duì)象序列化和反序列化機(jī)制。它具有跨平臺(tái)、對(duì)象圖形完整性和版本兼容性等優(yōu)點(diǎn),但也存在序列化后的字節(jié)流較大、可讀性差和性能較低等缺點(diǎn)。在使用時(shí)需要注意序列化 ID 的一致性、非序列化字段的處理和版本兼容性等問題。
以上就是JDK自帶的序列化方式優(yōu)缺點(diǎn)及實(shí)現(xiàn)原理面試精講的詳細(xì)內(nèi)容,更多關(guān)于JDK序列化方式的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
SpringBoot中MybatisX插件的簡(jiǎn)單使用教程(圖文)
MybatisX 是一款基于 IDEA 的快速開發(fā)插件,方便在使用mybatis以及mybatis-plus開始時(shí)簡(jiǎn)化繁瑣的重復(fù)操作,本文主要介紹了SpringBoot中MybatisX插件的簡(jiǎn)單使用教程,感興趣的可以了解一下2023-06-06SpringBoot實(shí)現(xiàn)api加密的示例代碼
在項(xiàng)目中,為了保證數(shù)據(jù)的安全,我們常常會(huì)對(duì)傳遞的數(shù)據(jù)進(jìn)行加密。本文主要介紹了SpringBoot實(shí)現(xiàn)api加密的示例代碼,感興趣的小伙伴們可以參考一下2021-07-07Java死鎖代碼實(shí)例及產(chǎn)生死鎖必備的四個(gè)條件
這篇文章主要介紹了Java死鎖代碼實(shí)例及產(chǎn)生死鎖必備的四個(gè)條件,Java 發(fā)生死鎖的根本原因是,在申請(qǐng)鎖時(shí)發(fā)生了交叉閉環(huán)申請(qǐng),synchronized在開發(fā)中最好不要嵌套使用,容易導(dǎo)致死鎖,需要的朋友可以參考下2024-01-01Java網(wǎng)絡(luò)編程之UDP網(wǎng)絡(luò)通信詳解
這篇文章主要為大家詳細(xì)介紹了Java網(wǎng)絡(luò)編程中的UDP網(wǎng)絡(luò)通信的原理與實(shí)現(xiàn),文中的示例代碼講解詳細(xì),具有一定的借鑒價(jià)值,需要的可以參考一下2022-09-09利用Spring Validation實(shí)現(xiàn)輸入驗(yàn)證功能
這篇文章主要給大家介紹了如何利用Spring Validation完美的實(shí)現(xiàn)輸入驗(yàn)證功能,文中有詳細(xì)的代碼示例,具有一定的參考價(jià)值,感興趣的朋友可以借鑒一下2023-06-06Java生產(chǎn)者消費(fèi)者模式實(shí)例分析
這篇文章主要介紹了Java生產(chǎn)者消費(fèi)者模式,結(jié)合實(shí)例形式分析了java生產(chǎn)者消費(fèi)者模式的相關(guān)組成、原理及實(shí)現(xiàn)方法,需要的朋友可以參考下2019-03-03SpringMVC異步處理操作(Callable和DeferredResult)
這篇文章主要介紹了SpringMVC異步處理操作(Callable和DeferredResult),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2021-01-01java實(shí)現(xiàn)簡(jiǎn)易版圖形界面計(jì)算器
這篇文章主要為大家詳細(xì)介紹了java實(shí)現(xiàn)簡(jiǎn)易版圖形界面計(jì)算器,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-05-05jedis獲取redis中二進(jìn)制圖片轉(zhuǎn)Base64方式
這篇文章主要介紹了jedis獲取redis中二進(jìn)制圖片轉(zhuǎn)Base64方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-07-07