全國(guó)咨詢(xún)/投訴熱線:400-618-4000

首頁(yè)技術(shù)文章正文

Java中怎樣將數(shù)據(jù)對(duì)象序列化和反序列化?

更新時(shí)間:2021-12-03 來(lái)源:黑馬程序員 瀏覽量:

IT培訓(xùn)班

程序在運(yùn)行過(guò)程中,可能需要將一些數(shù)據(jù)永久地保存到磁盤(pán)上,而數(shù)據(jù)在Java中都是保存在對(duì)象當(dāng)中的。那么我們要怎樣將對(duì)象中的數(shù)據(jù)保存到磁盤(pán)上呢?這時(shí)就需要使用Java中的對(duì)象序列化。


對(duì)象的序列化(Serializable)是指將一個(gè)Java對(duì)象轉(zhuǎn)換成一個(gè)I/O流中字節(jié)序列的過(guò)程。其目的是為了將對(duì)象保存到磁盤(pán)中,或允許在網(wǎng)絡(luò)中直接傳輸對(duì)象。對(duì)象序列化機(jī)制可以使內(nèi)存中的Java對(duì)象轉(zhuǎn)換成與平臺(tái)無(wú)關(guān)的二進(jìn)制流,既可以將這種二進(jìn)制流持久地保存在磁盤(pán)上,又可以通過(guò)網(wǎng)絡(luò)將這種二進(jìn)制流傳輸?shù)搅硪粋€(gè)網(wǎng)絡(luò)節(jié)點(diǎn),其他程序在獲得了這種二進(jìn)制流后,還可以將它恢復(fù)成原來(lái)的Java對(duì)象。這種將I/O流中的字節(jié)序列恢復(fù)為Java對(duì)象的過(guò)程被稱(chēng)之為反序列化(Deserialize)。

如果想讓某個(gè)對(duì)象支持序列化機(jī)制,那么這個(gè)對(duì)象所在的類(lèi)必須是可序列化的。在Java中,可序列化的類(lèi)必須實(shí)現(xiàn)Serializable 或Externalizable兩個(gè)接口之一。這兩個(gè)接口實(shí)現(xiàn)序列化機(jī)制的主要區(qū)別如表7-7所示。

與實(shí)現(xiàn)Serializable 接口相比,雖然實(shí)現(xiàn)Externalizable 接口可以帶來(lái)一定性能上的提升,但也將導(dǎo)致編程的復(fù)雜度增加。在實(shí)際開(kāi)發(fā)時(shí),大部分都是采用實(shí)現(xiàn)Serializable 接口的方式來(lái)實(shí)現(xiàn)序列化的。

表7-7 實(shí)現(xiàn) Serializable 與實(shí)現(xiàn)Externalizable 的對(duì)比

實(shí)現(xiàn) Serializable 接口實(shí)現(xiàn)Externalizable 接口
系統(tǒng)自動(dòng)存儲(chǔ)必要的信息由程序員決定所儲(chǔ)存的信息
Java內(nèi)部支持,易于實(shí)現(xiàn),只需實(shí)現(xiàn)該接口即可,不需要其他代碼支持接口中只提供了兩個(gè)空方法,實(shí)現(xiàn)該接口必須為兩個(gè)空方法提供實(shí)現(xiàn)
性能較差性能較好

 使用Serializable 接口實(shí)現(xiàn)序列化非常簡(jiǎn)單,只需要讓目標(biāo)類(lèi)實(shí)現(xiàn)Serializable 接口即可,無(wú)須實(shí)現(xiàn)任何方法。例如讓Person類(lèi)實(shí)現(xiàn)序列化接口的代碼如下:

public class Person implements Serializable {
    //為該類(lèi)指定一個(gè)serialVersionUID變量值
    private static final long serialVersionUID = 1 L:
        //聲明變量
        private int id;
    private String name;
    private int age;
    //此處省略各屬性的getter和setter方法
    ?
}

在上述代碼中,Person類(lèi)實(shí)現(xiàn)了Serializable接口,并指定了一個(gè)serialVersionUID變量值,該變量值的作用是標(biāo)識(shí)Java類(lèi)的序列化版本。如果不顯式地定義serialVersionUID變量值,那么將由JVM根據(jù)類(lèi)的相關(guān)信息計(jì)算出一個(gè)serialVersionUID變量值。

小提示:serialVersionUID適用于Java的序列化機(jī)制。簡(jiǎn)單來(lái)說(shuō),Java的序列化機(jī)制是通過(guò)判斷類(lèi)的serialVersionUID來(lái)驗(yàn)證版本一致性的。在進(jìn)行反序列化時(shí),JVM會(huì)把傳來(lái)的字節(jié)流中的serialVersionUID與本地相應(yīng)實(shí)體類(lèi)的serialVersionUID進(jìn)行比較,如果相同就認(rèn)為是一致的,可以進(jìn)行反序列化,否則就會(huì)出現(xiàn)序列化版本不一致的異常。因此,為了在反序列化時(shí)確保序列化版本的兼容性,最好在每一個(gè)要序列化的類(lèi)中加入privatestatic final long serialVersionUID的變量值,具體數(shù)值可自定義(默認(rèn)是1L,系統(tǒng)還可以根據(jù)類(lèi)名、接口名、成員方法及屬性等生成的一個(gè)64位的哈希字段)。這樣,某個(gè)對(duì)象被序列化之后,即使它所對(duì)應(yīng)的類(lèi)被修改了,該對(duì)象依然可以被正確地反序列化。






猜你喜歡:

Java中的NIO是什么?NIO的核心組成部分有哪些?

O和NIO的區(qū)別有哪些?NIO的優(yōu)點(diǎn)

Java的 io設(shè)計(jì)模式視頻教程

如何通過(guò)NIO實(shí)現(xiàn)群聊?【黑馬程序員】

黑馬程序員Java開(kāi)發(fā)培訓(xùn)

分享到:
在線咨詢(xún) 我要報(bào)名
和我們?cè)诰€交談!