更新時間:2021-12-03 來源:黑馬程序員 瀏覽量:
程序在運行過程中,可能需要將一些數(shù)據(jù)永久地保存到磁盤上,而數(shù)據(jù)在Java中都是保存在對象當中的。那么我們要怎樣將對象中的數(shù)據(jù)保存到磁盤上呢?這時就需要使用Java中的對象序列化。
對象的序列化(Serializable)是指將一個Java對象轉(zhuǎn)換成一個I/O流中字節(jié)序列的過程。其目的是為了將對象保存到磁盤中,或允許在網(wǎng)絡中直接傳輸對象。對象序列化機制可以使內(nèi)存中的Java對象轉(zhuǎn)換成與平臺無關的二進制流,既可以將這種二進制流持久地保存在磁盤上,又可以通過網(wǎng)絡將這種二進制流傳輸?shù)搅硪粋€網(wǎng)絡節(jié)點,其他程序在獲得了這種二進制流后,還可以將它恢復成原來的Java對象。這種將I/O流中的字節(jié)序列恢復為Java對象的過程被稱之為反序列化(Deserialize)。
如果想讓某個對象支持序列化機制,那么這個對象所在的類必須是可序列化的。在Java中,可序列化的類必須實現(xiàn)Serializable 或Externalizable兩個接口之一。這兩個接口實現(xiàn)序列化機制的主要區(qū)別如表7-7所示。
與實現(xiàn)Serializable 接口相比,雖然實現(xiàn)Externalizable
接口可以帶來一定性能上的提升,但也將導致編程的復雜度增加。在實際開發(fā)時,大部分都是采用實現(xiàn)Serializable 接口的方式來實現(xiàn)序列化的。
表7-7 實現(xiàn) Serializable 與實現(xiàn)Externalizable 的對比
實現(xiàn) Serializable 接口 | 實現(xiàn)Externalizable 接口 |
系統(tǒng)自動存儲必要的信息 | 由程序員決定所儲存的信息 |
Java內(nèi)部支持,易于實現(xiàn),只需實現(xiàn)該接口即可,不需要其他代碼支持 | 接口中只提供了兩個空方法,實現(xiàn)該接口必須為兩個空方法提供實現(xiàn) |
性能較差 | 性能較好 |
使用Serializable 接口實現(xiàn)序列化非常簡單,只需要讓目標類實現(xiàn)Serializable 接口即可,無須實現(xiàn)任何方法。例如讓Person類實現(xiàn)序列化接口的代碼如下:
public class Person implements Serializable { //為該類指定一個serialVersionUID變量值 private static final long serialVersionUID = 1 L: //聲明變量 private int id; private String name; private int age; //此處省略各屬性的getter和setter方法 ? }
在上述代碼中,Person類實現(xiàn)了Serializable接口,并指定了一個serialVersionUID變量值,該變量值的作用是標識Java類的序列化版本。如果不顯式地定義serialVersionUID變量值,那么將由JVM根據(jù)類的相關信息計算出一個serialVersionUID變量值。
小提示:serialVersionUID適用于Java的序列化機制。簡單來說,Java的序列化機制是通過判斷類的serialVersionUID來驗證版本一致性的。在進行反序列化時,JVM會把傳來的字節(jié)流中的serialVersionUID與本地相應實體類的serialVersionUID進行比較,如果相同就認為是一致的,可以進行反序列化,否則就會出現(xiàn)序列化版本不一致的異常。因此,為了在反序列化時確保序列化版本的兼容性,最好在每一個要序列化的類中加入privatestatic final long serialVersionUID的變量值,具體數(shù)值可自定義(默認是1L,系統(tǒng)還可以根據(jù)類名、接口名、成員方法及屬性等生成的一個64位的哈希字段)。這樣,某個對象被序列化之后,即使它所對應的類被修改了,該對象依然可以被正確地反序列化。