Java的serialization提供了一種持久化對象實例的機制。當持久化對象時,可能有一個特殊的對象數據成員,我們不想用serialization機制來保存它。 為了在一個特定對象的一個域上關閉serialization,可以在這個域前加上關鍵字transient。 當一個對象被序列化的時候,t ...
Java的serialization提供了一種持久化對象實例的機制。當持久化對象時,可能有一個特殊的對象數據成員,我們不想用serialization機制來保存它。
為了在一個特定對象的一個域上關閉serialization,可以在這個域前加上關鍵字transient。
當一個對象被序列化的時候,transient型變數的值不包括在序列化的表示中,然而非transient型的變數是被包括進去的。
簡而言之,被transient修飾的變數不參與序列化和反序列化。
接下來用代碼來證明一下。
新建一個Student類實現Serializable 介面,並重寫其toString方法便於觀察結果。
一個age屬性不被transient修飾,一個name屬性被transient修飾。
public class Student implements Serializable { private int age; private transient String name; public Student() { } public Student(int age, String name) { this.age = age; this.name = name; } @Override public String toString() { return "Student{" + "age=" + age + ", name='" + name + '\'' + '}'; } }
然後在TransientTest類裡邊測試。
PS:
為了代碼簡潔這裡的IO操作沒有進行try catch操作而是直接拋出了。
public class TestTransient { public static void main(String[] args) throws Exception { // 實例化一個Student對象. Student student = new Student(15, "HuaGe"); System.out.println(student); // 將student對象寫入磁碟文件(序列化) ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("student.txt")); oos.writeObject(student); oos.close(); // 從磁碟文件讀取student對象(反序列化) ObjectInputStream ois = new ObjectInputStream(new FileInputStream("student.txt")); student = (Student) ois.readObject(); System.out.println(student); } }
運行main方法,觀察控制台列印信息。
發現經過了序列化和反序列化後,name屬性從HuaGe變為了null.
這就說明瞭被transient修飾的變數不參與序列化和反序列化。
那有沒有例外吶?
我們知道,java中有兩種序列化的方式。
1. 實現Serializable介面。
2. 實現Externalizable介面。
Externalizable介面是Serializable介面的子類
源碼如下
public interface Externalizable extends java.io.Serializable { void writeExternal(ObjectOutput out) throws IOException; void readExternal(ObjectInput in) throws IOException, ClassNotFoundException; }
這個介面的兩個方法可以指定對類中的哪些屬性進行序列化。
使用這個介面時,無論屬性有沒有被transient修飾,
預設不對任何屬性進行序列化。所以實現了Externalizable介面的類
一般不再使用transient修飾屬性。
總結:
1. 被transient修飾的變數不參與序列化和反序列化
2. transient一般在實現了Serializable介面的類中使用。
以上只是個人的一些理解,如果哪裡不對,還請指出!