JAVA語言序列化和反序列化(1)-理解和總結

在JAVA中,一個大的應用程序需要保存很多對象的時候,由於虛擬機內存有限,(資源寶貴啊 )有時不可能所有有用的對象都放到內存中,因此,需要將不常用的對象暫時持久化的文件中,當需要這個對象時,再從文件把對象恢復到內存中,這就是所謂對象的序列化和反序列化。本文講實現如何將對象序列化到文件,然後再從文件反序列化到對象,你會發現其實特別簡單

先看一下對象的序列化和反序列化的關鍵是什麼

1,首先被序列化的對象必須實現 java.io.Serializable接口,其實這個接口沒定義任何方法

2,序列化時,需要用到對象輸出流ObjectOutputStream,然後通過文件輸出流構造ObjectOutputStream對象調用writeObject寫入到文件

3,反之,反序列化時用到對象輸入流ObjectIntputStream, 然後通過文件輸出流構造ObjectIntputStream對象調用readObject加載到內存,注意,是返回萬能Object類型

4,注意:序列化時transient變量(這個關鍵字的作用就是告知JAVA我不可以被序列化)和靜態變量不會被序列化(關於靜態變量不會序列化保留意見,本實例先不用靜態變量)

5,也是最應該注意的,如果你先序列化對象A後序列化B,那麼在反序列化的時候一定記着JAVA規定先讀到的對象是先被序列化的對象,不要先接收對象B,那樣會報錯


現在我們做一個實例,我們現在要序列化一個會員對象到文件裏,然後再把它讀出來

先創建會員類

import java.io.*;
import java.util.*;

//一定要實現Serializable接口才能被序列化
public class UserInfo implements Serializable {
public String userName;
public String userPass;
//注意,userAge變量前面的transient
public transient int userAge;

public UserInfo(){
}

public UserInfo(String username,String userpass,int userage){
 this.userName=username;
 this.userPass=userpass;
 this.userAge=userage;
}

public String toString(){
 return "用戶名: "+this.userName+";密碼:"+this.userPass+
  ";年齡:"+this.userAge;
}
}
接着我們開始寫如何序列化和反序列化

import java.io.*;
import java.util.*;
public class Test {

//序列化對象到文件
public static void serialize(String fileName){
 try
 {
  //創建一個對象輸出流,講對象輸出到文件
  ObjectOutputStream out=new ObjectOutputStream(new FileOutputStream(fileName));

  out.writeObject("序列化日期是:");  //序列化一個字符串到文件

  out.writeObject(new Date()); //序列化一個當前日期對象到文件

  UserInfo user=new UserInfo("renyanwei","888888",20);
  out.writeObject(user); //序列化一個會員對象

  out.close();
 }
 catch (Exception x)
 {
  System.out.println(x.toString());
 }

}
//從文件反序列化到對象
public static void deserialize(String fileName){
 try
 {
  //創建一個對象輸入流,從文件讀取對象
  ObjectInputStream in=new ObjectInputStream(new FileInputStream(fileName));

  //注意讀對象時必須按照序列化對象順序讀,否則會出錯
  //讀取字符串
  String today=(String)(in.readObject());
  System.out.println(today);

  //讀取日期對象
  Date date=(Date)(in.readObject());
  System.out.println(date.toString());

  //讀取UserInfo對象並調用它的toString()方法
  UserInfo user=(UserInfo)(in.readObject());  
  System.out.println(user.toString());

  in.close();
 }
 catch (Exception x)
 {
  System.out.println(x.toString());
 }

}

public static void main(String[] args) {
 serialize("D:\\test.txt");
 System.out.println("序列化完畢");

 deserialize("D:\\test.txt");
 System.out.println("反序列化完畢");
}

}

運行結果:


序列化完畢
今天是:
Thu Oct 23 18:31:11 CST 2008
用戶名: renyanwei;密碼:888888;年齡:0
反序列化完畢

我們看代碼總結一下,serialize方法實現了對象的序列化功能,講對象序列化到文件,使用了文件的輸出流ObjectOutputStream的writerObject方法一次寫入一個對象,對象在文件中按寫入順序存儲。

deserialize方法實現反序列化,從文件將對象反序列化到內存,使用了ObjectIntputStream的readObject方法,因爲序列化對象是按照寫入順序存儲的,所以在反序列化時必須按照寫入順序讀取

大家看了結果就應該知道了,標明transit的實例變量是沒被序列化進去的


其實序列化的意義不止如此,不但可以將對象放到文件流,還可以放到網絡流裏,比如我們做的Socket程序,你不但可以發送給對方一行消息,還可以發送過去一個對象,我估計QQ就不止是把我們的消息發送到對方,應該是一個對象,裏面包含了發送者的信息什麼的


發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章