File 類的介紹:http://www.cnblogs.com/ysocean/p/6851878.htmlhtml
Java IO 流的分類介紹:http://www.cnblogs.com/ysocean/p/6854098.htmljava
Java IO 字節輸入輸出流:http://www.cnblogs.com/ysocean/p/6854541.html服務器
Java IO 字符輸入輸出流:https://i.cnblogs.com/EditPosts.aspx?postid=6859242網絡
Java IO 包裝流:http://www.cnblogs.com/ysocean/p/6864080.html數據結構
一、什麼是序列化與反序列化?分佈式
序列化:指把堆內存中的 Java 對象數據,經過某種方式把對象存儲到磁盤文件中或者傳遞給其餘網絡節點(在網絡上傳輸)。這個過程稱爲序列化。通俗來講就是將數據結構或對象轉換成二進制串的過程ide
反序列化:把磁盤文件中的對象數據或者把網絡節點上的對象數據,恢復成Java對象模型的過程。也就是將在序列化過程當中所生成的二進制串轉換成數據結構或者對象的過程post
二、爲何要作序列化?this
①、在分佈式系統中,此時須要把對象在網絡上傳輸,就得把對象數據轉換爲二進制形式,須要共享的數據的 JavaBean 對象,都得作序列化。htm
②、服務器鈍化:若是服務器發現某些對象很久沒活動了,那麼服務器就會把這些內存中的對象持久化在本地磁盤文件中(Java對象轉換爲二進制文件);若是服務器發現某些對象須要活動時,先去內存中尋找,找不到再去磁盤文件中反序列化咱們的對象數據,恢復成 Java 對象。這樣能節省服務器內存。
三、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
private String name;//須要序列化 transient private int age;//不須要序列化
那麼咱們在反序列化的時候,打印出來的就是Person [name=vae, age=0],整型數據默認值爲 0
問題2:序列化版本問題,在完成序列化操做後,因爲項目的升級或修改,可能咱們會對序列化對象進行修改,好比增長某個字段,那麼咱們在進行反序列化就會報錯:
解決辦法:在 JavaBean 對象中增長一個 serialVersionUID 字段,用來固定這個版本,不管咱們怎麼修改,版本都是一致的,就能進行反序列化了
private static final long serialVersionUID = 8656128222714547171L;