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