什么是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:
對(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