No.22 對(duì)象流:對(duì)象序列化/反序列化

什么是Java對(duì)象序列化:

  • Java平臺(tái)允許我們?cè)趦?nèi)存中創(chuàng)建可復(fù)用的Java對(duì)象,但一般情況下,只有當(dāng)JVM處于運(yùn)行時(shí),這些對(duì)象才可能存在,即這些對(duì)象的生命周期不會(huì)比JVM的生命周期更長(zhǎng)。
  • 在現(xiàn)實(shí)應(yīng)用中,就可能要求在JVM停止運(yùn)行之后能夠保存(持久化)指定的對(duì)象,并在將來(lái)重新讀取被保存的對(duì)象。Java對(duì)象序列化就能夠幫助我們實(shí)現(xiàn)該功能。
  • 除了在持久化對(duì)象時(shí)會(huì)用到對(duì)象序列化之外,當(dāng)使用RMI(遠(yuǎn)程方法調(diào)用),或在網(wǎng)絡(luò)中傳遞對(duì)象時(shí),都會(huì)用到對(duì)象序列化。
  • 使用Java對(duì)象序列化,在保存對(duì)象時(shí),會(huì)把其狀態(tài)保存為一組字節(jié),在未來(lái),再將這些字節(jié)組裝成對(duì)象。必須注意地是,對(duì)象序列化保存的是對(duì)象的"狀態(tài)",即它的成員變量,對(duì)象序列化不會(huì)關(guān)注類(lèi)中的靜態(tài)變量。

怎么進(jìn)行對(duì)象序列化:

(1)Java中,只要一個(gè)類(lèi)實(shí)現(xiàn)了Serializable接口,那么它就可以被序列化;
(2)如果一個(gè)類(lèi)中有些字段不希望被序列化,例如如果一個(gè)用戶有一些敏感信息(如密碼),為了安全起見(jiàn),不想被記錄在文件中被傳輸,這些信息對(duì)應(yīng)的變量加上transient關(guān)鍵字就可以了; 序列化對(duì)象的時(shí)候,這個(gè)屬性就不會(huì)序列化到指定的目的地中(file中)。

使用ObjectOutputStream將對(duì)象序列化/ObjectInputStream將對(duì)象反序列化:

  • java.io中的類(lèi)ObjectInputStream 和ObjectOutputStream是高層次的數(shù)據(jù)流,它們包含序列化和反序列化對(duì)象的方法。他們呢是InputStream/OutputStream的直接子類(lèi)。

  • 例子:
    首先我們自定義一個(gè)類(lèi):

    private static final long serialVersionUID = 1L;
    private String name;
    private int age;
    private transient int password;
    private Gender gender;
    
    public Person(String name, int age, int password, Gender gender) {
      super();
      this.name = name;
      this.age = age;
      this.password = password;
      this.gender = gender;
    }
    
    @Override
    public String toString() {
      String str = "[" + "name : " + name + "\r\n" +
                 "age : " + age + "\r\n" + 
                "password :" + password + "\r\n"
              + "gender : " + gender + "]";
          return str;
    }
    

其中Gender屬性是一個(gè)enum(枚舉)類(lèi),查閱API文檔我們知道enum實(shí)現(xiàn)了Serialable接口所以該屬性也可以被序列化;
enum Gender {MAIL,FEMAIL}

下面我們使用ObjectOutputStream和ObjectInputStream實(shí)現(xiàn)序列化反序列化;

public class Test {
public static void main(String[] args) {
    String fileName = "D://Person.ser";
    File file = new File(fileName);

    ObjectInputStream ois = null;
    ObjectOutputStream oos = null;

    Person person = new Person("lisi", 16, 1111111, Gender.MAIL);
    try {
        oos = new ObjectOutputStream(new FileOutputStream(file));
        oos.writeObject(person);
        oos.close();
        
        
        ois = new ObjectInputStream(new FileInputStream(file));
        Object p= ois.readObject();
        ois.close();
        System.out.println(p);
    } catch (IOException e) {
        e.printStackTrace();
    } catch (ClassNotFoundException e) {
        e.printStackTrace();
    }

  }
}

serialVersionUID字段意義:

在實(shí)際應(yīng)用中我們常遇到要修改Person類(lèi)的情況,由于Person之前已經(jīng)實(shí)現(xiàn)了Serialable接口,如果在序列化之后,Person這個(gè)類(lèi)發(fā)生了改變,比如,多了一個(gè)成員變量。我們經(jīng)過(guò)試驗(yàn)可以得到這樣的結(jié)果:

Exception in thread “main” java.io.InvalidClassException: Person; local class incompatible: stream classdesc serialVersionUID = xxxxxxxx, local class serialVersionUID = yyyyyyyyyy ;

意思就是說(shuō),文件流中的class和classpath中的class,也就是修改過(guò)后的class,不兼容了,處于安全機(jī)制考慮,程序拋出了錯(cuò)誤,并且拒絕載入。

出現(xiàn)這樣的結(jié)果就是,我們?nèi)绻恢恍蛄谢膶?duì)象的serialVersionUID字段時(shí)候,java會(huì)給我們自動(dòng)生成一個(gè)serialVersionUID,帶改變后又給我們生成了另一個(gè)serialVersionUID,導(dǎo)致了反序列化失敗。

此時(shí)我們就需要對(duì)該類(lèi)自己指定一個(gè)serialVersionUID值,指定后再修改就不會(huì)出現(xiàn)上述問(wèn)題了。(當(dāng)我們使用了如Eclipse這樣的編譯器時(shí),編譯器會(huì)提示我們生成該字段,不生成的話就會(huì)報(bào)出警告)

參考資料:

transient:

http://www.cnblogs.com/lanxuezaipiao/p/3369962.html

對(duì)象序列化為何要定義serialVersionUID的來(lái)龍去脈

http://lenjey.iteye.com/blog/513736)http://lenjey.iteye.com/blog/513736

理解Java對(duì)象序列化

http://www.blogjava.net/jiangshachina/archive/2012/02/13/369898.html

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書(shū)系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容