1. 程式人生 > >java io詳解六:序列化與反序列化(物件流)

java io詳解六:序列化與反序列化(物件流)

1、什麼是序列化與反序列化?

  序列化:指把堆記憶體中的 Java 物件資料,通過某種方式把物件儲存到磁碟檔案中或者傳遞給其他網路節點(在網路上傳輸)。這個過程稱為序列化。通俗來說就是將資料結構或物件轉換成二進位制串的過程

  反序列化:把磁碟檔案中的物件資料或者把網路節點上的物件資料,恢復成Java物件模型的過程。也就是將在序列化過程中所生成的二進位制串轉換成資料結構或者物件的過程

 

2、為什麼要做序列化?

  ①、在分散式系統中,此時需要把物件在網路上傳輸,就得把物件資料轉換為二進位制形式,需要共享的資料的 JavaBean 物件,都得做序列化。

  ②、伺服器鈍化:如果伺服器發現某些物件好久沒活動了,那麼伺服器就會把這些記憶體中的物件持久化在本地磁碟檔案中(Java物件轉換為二進位制檔案);如果伺服器發現某些物件需要活動時,先去記憶體中尋找,找不到再去磁碟檔案中反序列化我們的物件資料,恢復成 Java 物件。這樣能節省伺服器記憶體。

 

3、Java 怎麼進行序列化?

  ①、需要做序列化的物件的類,必須實現序列化介面:Java.lang.Serializable 介面(這是一個標誌介面,沒有任何抽象方法),Java 中大多數類都實現了該介面,比如:String,Integer

  ②、底層會判斷,如果當前物件是 Serializable 的例項,才允許做序列化,Java物件 instanceof Serializable 來判斷。

  ③、在 Java 中使用物件流來完成序列化和反序列化

    ObjectOutputStream:通過 writeObject()方法做序列化操作

    ObjectInputStream

:通過 readObject() 方法做反序列化操作

    

 

 

 第一步:建立一個 JavaBean 物件

public class Person implements Serializable{
    private String name;
    private int age;
     
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public int getAge() {
        return age;
    }
    public void setAge(int age) {
        this.age = age;
    }
    @Override
    public String toString() {
        return "Person [name=" + name + ", age=" + age + "]";
    }
    public Person(String name, int age) {
        super();
        this.name = name;
        this.age = age;
    }
}  

 第二步:使用 ObjectOutputStream 物件實現序列化

//在根目錄下新建一個 io 的資料夾
        OutputStream op = new FileOutputStream("io"+File.separator+"a.txt");
        ObjectOutputStream ops = new ObjectOutputStream(op);
        ops.writeObject(new Person("vae",1));
         
        ops.close();

我們開啟 a.txt 檔案,發現裡面的內容亂碼,注意這不需要我們來看懂,這是二進位制檔案,計算機能讀懂就行了。

錯誤一:如果新建的 Person 物件沒有實現 Serializable 介面,那麼上面的操作會報錯:

    

第三步:使用ObjectInputStream 物件實現反序列化

  反序列化的物件必須要提供該物件的位元組碼檔案.class

InputStream in = new FileInputStream("io"+File.separator+"a.txt");
        ObjectInputStream os = new ObjectInputStream(in);
        byte[] buffer = new byte[10];
        int len = -1;
        Person p = (Person) os.readObject();
        System.out.println(p);  //Person [name=vae, age=1]
        os.close();

問題1:如果某些資料不需要做序列化,比如密碼,比如上面的年齡?

解決辦法:在欄位面前加上 transient

1

2

private String name;//需要序列化

transient private int age;//不需要序列化

  那麼我們在反序列化的時候,打印出來的就是Person [name=vae, age=0],整型資料預設值為 0 

 

問題2:序列化版本問題,在完成序列化操作後,由於專案的升級或修改,可能我們會對序列化物件進行修改,比如增加某個欄位,那麼我們在進行反序列化就會報錯:

 

 

解決辦法:在 JavaBean 物件中增加一個 serialVersionUID 欄位,用來固定這個版本,無論我們怎麼修改,版本都是一致的,就能進行反序列化了

1

private static final long serialVersionUID = 8656128222714547171L;