Java筆記-I/O流之物件序列化流
物件序列化流
物件序列化流基本介紹
使用工具:ObjectOutputStream
,ObjectInputStream
介紹:將物件以檔案的形式儲存在硬碟中,使之能更方便的傳輸。
條件:必須實現Serializable介面(實現了這個介面,但並不需要重寫任何方法)
物件序列化:將物件儲存到本地檔案
ObjectOutputStream
:本身有寫基本資料型別和引用資料型別的方法,還有寫物件的方法 writeObject()
物件反序列化:將檔案中儲存的物件讀取回來
ObjectInputStream
:基本等同於上邊的OOS
物件序列化和反序列化的含義:
將記憶體中的一個物件以位元組碼的形式永久儲存到平臺的檔案中去。
就算java程式的記憶體關閉,檔案還在等下一次java程式執行,從本地檔案再一次讀取回來儲存的物件,重新開闢一塊空間
儲存的是之前儲存的物件的所有資料。
如何序列化一組物件:
問:一個檔案只能序列化一個物件,那麼如果多個物件呢?
答:將多個物件放入物件陣列,將物件陣列儲存到本地檔案
注意:
1.異常:NotSerializableException
不是序列化介面異常:一個類的物件要想被序列化到檔案,那麼該物件對應的類必須實現序列化介面。
2.用static修飾的屬性無法被序列化到本地檔案,方法也是一樣,所以:序列化只是儲存了物件堆記憶體中的資料。
3.如果某一個屬性不想被序列化,應該如何處理?
(1).用static修飾屬性
(2).transient
關鍵字修飾的屬性無法被序列化。推薦使用
4.異常:InvalidClassException
類無效異常:
demoObjectStream.Person;
local class incompatible: stream classdesc serialVersionUID = -3834665631564771279,
local class serialVersionUID = -2858643162280623880
serialVersionUID
是序列化號:
是根據當前類的各項屬性動態計算出來,為了標識當前儲存到本地檔案的物件
導致問題的原因:
物件序列化以後,沒有馬上讀回來。
或者說,讀回來之前該物件對應的類屬性結構發生了改變,比如:屬性個數或屬性名發生了改變。
那麼stream classdesc serialVersionUID
也會對應發生改變
那麼跟本地檔案儲存的local class serialVersionUI
對應不上,就無法讀取回來
如何解決: 看到類名上面的警告線了麼? 按提示自動生成即可 1.ADD Default…ID 無視當前屬性結構,直接寫一個死的ID 2.根據當前屬性結構動態分配一個數
這兩種沒區別,只要寫上一個死的ID就可以,然後將這個ID修改成:異常提示中本地儲存物件的serialVersionUI
即可。
5.如果一個物件的屬性裡面包含其他引用資料型別,那麼該引用資料型別也要實現序列化介面。
序列化演示程式碼:
為了一次演示的比較全:所以建三個類:
1.Person.java 2.Father.java 3.ObjectStream.java
1.Person.java類:
import java.io.Serializable;
//實現Serializable介面
public class Person implements Serializable {
private int age;
private String name;
// 不想被序列化的sex屬性 : 用 transient 修飾 ,與private與否無關
private transient String sex;
// 如果一個物件的屬性裡面包含其他引用資料型別,那麼該引用資料型別也要實現序列化介面
private Father father;
//帶參構造方法,由於我沒宣告無參構造,所以不能使用無參構造了,想用的話就加上
public Person(int age, String name, String sex, Father father) {
super();
this.age = age;
this.name = name;
this.sex = sex;
this.father = father;
}
//重寫toString方法
@Override
public String toString() {
return "Person [age=" + age + ", name=" + name + ", sex=" + sex + ", father=" + father + "]";
}
//setter getter 方法
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getSex() {
return sex;
}
public void setSex(String sex) {
this.sex = sex;
}
public Father getFather() {
return father;
}
public void setFather(Father father) {
this.father = father;
}
}
2.Father.java類:
import java.io.Serializable;
//實現Serializable介面
public class Father implements Serializable {
String name;
public Father(String name) {
super();
this.name = name;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
@Override
public String toString() {
return "Father [name=" + name + "]";
}
}
3.ObjectStream.java類:
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
//執行順序
//1.執行序列化-->將物件陣列存到本地檔案中
//2.執行反序列化-->從本地檔案中讀取到存入的物件陣列
public class ObjectStream {
// 物件反序列化
public static void main(String[] args) {
try {
ObjectInputStream ois = new ObjectInputStream(new FileInputStream("D:\\a.txt"));
Person[] p = (Person[]) ois.readObject();
for (Person per : p) {
System.out.println(per);
}
} catch (IOException e) {
e.printStackTrace();
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
}
// 物件序列化
public static void main1(String[] args) {
Person[] p = { new Person(11, "小明", "男", new Father("大明")), new Person(11, "小張", "男", new Father("大張")),
new Person(11, "小芳", "女", new Father("大芳")), new Person(11, "小美", "女", new Father("大美")) };
try {
ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("D:\\a.txt"));
oos.writeObject(p);
oos.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}