在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就不止是把我們的消息發送到對方,應該是一個對象,裏面包含了發送者的信息什麼的