java物件實現Serializable介面(整理)
Serializable 在工作中很少用到 之前也懂一些 今天特意整理一下
在還沒有深入瞭解serializable介面之前,像很多程式設計師一樣,以為一個物件實現serializable介面就被序列化了。 最近在接觸ehcache快取的時候,將物件快取起來,該物件需要先實現Serializable介面,然而,我們會發現物件並沒有真正的被序列化。 下面讓我們一起來總結一下Serializable介面的實現原理。 當一個類實現了Seializable介面(該介面僅為標記介面,不包含任何方法定義),表示該類可以序列化,序列化的目的是將一個實現了Serializable介面的物件可以轉換成一個位元組序列,儲存物件的狀態。- package org.test.domain;
- import java.io.Serializable;
- publicclass Person implements Serializable{
- /**
- *
- */
- privatestaticfinallong serialVersionUID = 1L;
- protected String name;
- protectedtransientint age;
- public Person(){}
- public Person(String name,int age){
- this.name = name;
- this.age = age;
- }
- public String getName() {
- return name;
- }
- publicvoid setName(String name) {
- this.name = name;
- }
- publicint getAge() {
- return age;
- }
- publicvoid setAge(int age) {
- this.age = age;
- }
- public String toString()
- {
- return"this is person:"+"name:"+this.name+"——age:"+this.age;
- }
- }
User類
- package org.test.domain;
- import java.io.Serializable;
- publicclass User extends Person{
- /**
- *
- */
- privatestaticfinallong serialVersionUID = 1L;
- private String name;
- private String password;
- public User() {
- }
- public User(String name,String password,int age)
- {
- this.name = name;
- this.password = password;
- this.age = age;
- }
- public String getName() {
- return name;
- }
- publicvoid setName(String name) {
- this.name = name;
- }
- public String getPassword() {
- return password;
- }
- publicvoid setPassword(String password) {
- this.password = password;
- }
- public String toString()
- {
- return"this is user:"+"name:"+this.name+"——password:"+this.password+"——age:"+this.age;
- }
- }
SerializableTest類
- package org.test.main;
- import java.io.ByteArrayInputStream;
- import java.io.ByteArrayOutputStream;
- import java.io.IOException;
- import java.io.ObjectInputStream;
- import java.io.ObjectOutputStream;
- import org.test.domain.Person;
- import org.test.domain.User;
- publicclass SerializableTest {
- publicstaticvoid main(String[] args) {
- Person p1 = (Person)deSerialByte(serialByte(new User("user","1234",15)));
- //Person p2 = (Person)deSerialByte(serialByte(new Person("person",10)));
- System.out.println("p1:"+p1.toString());
- //System.out.println("p2:"+p2.toString());
- }
- //序列化一個物件(可以儲存到一個檔案也可以儲存到位元組陣列)這裡儲存到自己陣列
- publicstaticbyte[] serialByte(Object obj)
- {
- ByteArrayOutputStream baos = new ByteArrayOutputStream();
- ObjectOutputStream oos;
- try {
- oos = new ObjectOutputStream(baos);
- oos.writeObject(obj);
- oos.close();
- return baos.toByteArray();
- } catch (IOException e) {
- thrownew RuntimeException(e.getMessage());
- }
- }
- //反序列化一個物件
- publicstatic Object deSerialByte(byte[] by)
- {
- ObjectInputStream ois;
- try {
- ois = new ObjectInputStream(new ByteArrayInputStream(by));
- return ois.readObject();
- } catch (Exception e) {
- thrownew RuntimeException(e.getMessage());
- }
- }
- }
serialVersionUID作用:
序列化時為了保持版本的相容性,即在版本升級時反序列化仍保持物件的唯一性。
有兩種生成方式:
一個是預設的1L,比如:private static final long serialVersionUID = 1L;
一個是根據類名、介面名、成員方法及屬性等來生成一個64位的雜湊欄位,比如:
private static final long serialVersionUID = xxxxL;
當你一個類實現了Serializable介面,如果沒有定義serialVersionUID,Eclipse會提供這個提示功能告訴你去定義 。在Eclipse中點選類中warning的圖示一下,Eclipse就會自動給定兩種生成的方式。如果不想定義它,在Eclipse的設定中也
可以把它關掉的,設定如下:
Window ==> Preferences ==> Java ==> Compiler ==> Error/Warnings ==>
Potential programming problems
將Serializable class without serialVersionUID的warning改成ignore即可。
如果你沒有考慮到相容性問題時,就把它關掉,不過有這個功能是好的,只要任何類別實現了Serializable這個介面的話,如果沒有加入serialVersionUID,Eclipse都會給你warning提示,這個serialVersionUID為了讓該類別Serializable向後相容。
如果你的類Serialized存到硬碟上面後,可是後來你卻更改了類別的field(增加或減少或改名),當你Deserialize時,就會出現Exception的,這樣就會造成不相容性的問題。
但當serialVersionUID相同時,它就會將不一樣的field以type的預設值Deserialize,可避開不相容性問題。
注意以下幾點:
1、若繼承的父類沒有實現Serializable介面,但是又想讓子類可序列化,子類實現Serializable介面,子類必須有可訪問的無參構造方法,用於儲存和恢復父類的public或protected或同包下的package欄位的狀態,否則在序列化或反序列化時會丟擲RuntimeException異常,對於序列化後的子類,在進行反序列化時,理論上無法初始化父類中private(不可訪問)物件變數的狀態或值。
2、在對可序列化類中的屬性進行序列化時,如果遇到不可序列化的物件變數,此時會針對不可序列化的類丟擲NotSerializableException異常
3、對於可序列化的非陣列類,強烈建議顯示宣告static型、long型、final型serialVersionUID欄位用於標識當前序列化類的版本號,否則在跨作業系統、跨編譯器之間進行序列化和反序列化時容易出現InvalidClassException異常