天天看點

Java中序列化的serialVersionUID作用

Java序列化是将一個對象編碼成一個位元組流,反序列化将位元組流編碼轉換成一個對象。 序列化是Java中實作持久化存儲的一種方法;為資料傳輸提供了線路級對象表示法。

Java的序列化機制是通過在運作時判斷類的serialVersionUID來驗證版本一緻性的。在進行反序列化時,JVM會把傳來的位元組流中的serialVersionUID與本地相應實體(類)的serialVersionUID進行比較,如果相同就認為是一緻的,可以進行反序列化,否則就會出現序列化版本不一緻的異常。

Eclipse中The serializable class XXXXXX does not declare a static final serialVersionUID field of type long出現這樣的警告處理辦法。

當采用程式的Add default Serial version ID修複時,Eclipse會加上:private static final long serialVersionUID = 1L;

當采用程式的Add generated Serial version ID修複時,Eclipse會加上:private static final long serialVersionUID = xxxxL;

其實這個問題出現的具體原因是和序列化中的這個serialVersionUID有關。 serialVersionUID 用來表明類的不同版本間的相容性。有兩種生成方式: 一個是預設的1L;另一種是根據類名、接口名、成員方法及屬性等來生成一個64位的哈希字段 。

在JDK中,可以利用JDK的bin目錄下的serialver.exe工具産生這個serialVersionUID 的值,對于Test.class,執行指令:

serialver Test 這時JVM(java虛拟機)會生成一個哈希字段。

對比一下這個哈希字段的值與方法2中生成的字段值是一樣的,可見,在CMD中使用serialver指令就是根據類名、接口名、成員方法及屬性等來生成哈希字段的。

java類中為什麼需要重載 serialVersionUID 屬性。

當兩個程序在進行遠端通信時,彼此可以發送各種類型的資料。無論是何種類型的資料,都會以二進制序列的形式在網絡上傳送。發送方需要把這個Java對象轉換為位元組序列,才能在網絡上傳送;接收方則需要把位元組序列再恢複為Java對象。 把Java對象轉換為位元組序列的過程稱為對象的序列化,把位元組序列恢複為Java對象的過程稱為對象的反序列化。

  對象的序列化主要有兩種用途:

  1) 把對象的位元組序列永久地儲存到硬碟上,通常存放在一個檔案中;

  2) 在網絡上傳送對象的位元組序列。

java.io.ObjectOutputStream代表對象輸出流,它的writeObject(Object obj)方法可對參數指定的obj對象進行序列化,把得到的位元組序列寫到一個目标輸出流中。 java.io.ObjectInputStream代表對象輸入流,它的readObject()方法從一個源輸入流中讀取位元組序列,再把它們反序列化為一個對象,并将其傳回。 隻有實作了Serializable或Externalizable接口的類的對象才能被序列化。Externalizable接口繼承自Serializable接口,實作Externalizable接口的類完全由自身來控制序列化的行為,而僅實作Serializable接口的類可以采用預設的序列化方式 。 凡是實作Serializable接口的類都有一個表示序列化版本辨別符的靜态變量:private static final long serialVersionUID; 類的serialVersionUID的預設值完全依賴于Java編譯器的實作,對于同一個類,用不同的Java編譯器編譯,有可能會導緻不同的serialVersionUID。顯式地定義serialVersionUID有兩種用途:

  1)在某些場合,希望類的不同版本對序列化相容,是以需要確定類的不同版本具有相同的serialVersionUID;在某些場合,不希望類的不同版本對序列化相容,是以需要確定類的不同版本具有不同的serialVersionUID。

  2)當你序列化了一個類執行個體後,希望更改一個字段或添加一個字段,不設定serialVersionUID,所做的任何更改都将導緻無法反序化舊有執行個體,并在反序列化時抛出一個異常。如果你添加了serialVersionUID,在反序列舊有執行個體時,新添加或更改的字段值将設為初始化值(對象為null,基本類型為相應的初始預設值),字段被删除将不設定。

序列化算法一般會按步驟:

将對象執行個體相關的類中繼資料輸出。

遞歸地輸出類的超類描述直到不再有超類。

類中繼資料完了以後,開始從最頂層的超類開始輸出對象執行個體的實際資料值。

從上至下遞歸輸出執行個體的資料

引自:http://blog.csdn.net/linghu_java/article/details/25194821