序列化----Serializable---是將對象轉(zhuǎn)換成二進(jìn)制數(shù)據(jù)保存到本地磁盤或者通過網(wǎng)絡(luò)進(jìn)行傳輸,這個大家都知道。
一個對象在序列化是包括兩部分:
1、類描述信息,包括包路徑、繼承關(guān)系、訪問權(quán)限控制、變量描述、方法參數(shù)及返回值,但是不記錄方法、構(gòu)造、靜態(tài)變量的具體實現(xiàn);
2、非瞬態(tài)(transient關(guān)鍵字)和非靜態(tài)(static關(guān)鍵字)的變量。
關(guān)于transient,http://blog.csdn.net/chmingyuan/article/details/51252416
首先有幾個問題,不知你是否也思考過,當(dāng)我們的一個javabean實現(xiàn)了serializable接口后,IDE會提示沒有聲明serial versionID
1、為什么對象在序列化時要顯式聲明一個serial versionID?
2、對象序列化后,再反序列化時,對象的屬性值是否可以修改?
第一個問題:
SerialVersionUID也叫做流標(biāo)識符,描述的是類的版本定義,如果我們不顯式的聲明,編譯器會在編譯的時候自動生成,生成是根據(jù)包名、類名、繼承關(guān)系、非私有的方法和屬性及參數(shù)、返回值等計算出來的,基本上保證了其唯一性。
SerialVersionUID的作用呢,JVM在反序列化時,會比較數(shù)據(jù)流中的SerialVersionUID與類中的SerialVersionUID是否相同,如果相同,則認(rèn)為類沒有發(fā)生改變,可以把數(shù)據(jù)流反序列化為對象;如果不相同,則會拋出異常。
下面用代碼說明
public static void main(String[] args) { Person person=new Person(); person.setName("張三豐"); SerializableUtils.writeObject(person); }現(xiàn)在有一個普通的類,且沒有顯式聲明SerialVersionUID
class Person implements Serializable{ PRivate String name; public String getName() { return name; } public void setName(String name) { this.name = name; }}這里寫了一個工具class SerializableUtils{ private static final String FILE_NAME="d:/test.txt"; //序列化 public static void writeObject(Serializable seria){ try { ObjectOutputStream out=new ObjectOutputStream(new FileOutputStream(FILE_NAME)); out.writeObject(seria); out.close(); } catch (Exception e) { e.printStackTrace(); } } //反序列化 public static Object readObject(){ Object obj=null; try { ObjectInputStream in=new ObjectInputStream(new FileInputStream(FILE_NAME)); obj = in.readObject(); in.close(); } catch (Exception e) { e.printStackTrace(); } return obj; }} 此時序列化,反序列化都是沒有問題的。但在在分布式應(yīng)用中隨時都有可能出現(xiàn)新增屬性的情況,例如我這里要給Persong對象新增一個age屬性,但是對象已經(jīng)序列化在網(wǎng)絡(luò)中傳輸了,此時如果將數(shù)據(jù)流反序列化,會拋出異常class Person implements Serializable{ private String name; private int age;反序列化: Person obj = (Person)SerializableUtils.readObject(); System.out.println(obj);java.io.InvalidClassException: serializable.Person; local class incompatible: stream classdesc serialVersionUID = 4178445814915777397, local class serialVersionUID = 5875760287446429789 因為本地的類發(fā)生了變化,類的版本對應(yīng)不上了,但是我們可以通過顯式的聲明類的版本serialVersionUID,告訴JVM我的類沒有發(fā)生變化,反序列時便不會再拋出異常。
新聞熱點(diǎn)
疑難解答