transient關鍵字和@Transient 注解


版權聲明:本文為CSDN博主「wolfshadow.cn」的原創文章,遵循 CC 4.0 BY-SA 版權協議,轉載請附上原文出處鏈接及本聲明。
原文鏈接:https://blog.csdn.net/u010188178/article/details/83581506

 

百度百科的解釋:

        Java語言的關鍵字,變量修飾符,如果用transient聲明一個實例變量,當對象存儲時,它的值不需要維持。換句話來說就是,用transient關鍵字標記的成員變量不參與序列化過程。

作用:
        Java的serialization提供了一種持久化對象實例的機制。當持久化對象時,可能有一個特殊的對象數據成員,我們不想用serialization機制來保存它。為了在一個特定對象的一個域上關閉serialization,可以在這個域前加上關鍵字transient。當一個對象被序列化的時候,transient型變量的值不包括在序列化的表示中,然而非transient型的變量是被包括進去的。


 

編碼試驗加以證明:

1.自定義類(為了方便,我直接在main方法所在類中添加的一個靜態屬性類,或者叫成員類

public static class TransientTest implements Serializable{
        private static final long serialVersionUID = 233858934995755239L;
        private String name1;
        private transient String name2;
        
        public TransientTest(String name1,String name2){
            this.name1 = name1;
            this.name2 = name2;
        }
        public String toString(){
            return String.format("TransientTest.toString(): name1=%s,name2=%s", name1,name2);
        }
    }

2.寫一個測試方法

public static void testTransient(){
        String name1="常規屬性",name2="transient修飾的屬性";
        TransientTest test = new TransientTest(name1, name2);
        System.out.println("序列化前:"+test.toString());
        ObjectOutputStream outStream;
        ObjectInputStream inStream;
        String filePath = "D:/test/object/TransientTest.obj";
        try {
            outStream = new ObjectOutputStream(new FileOutputStream(filePath));
            outStream.writeObject(test);
            
            inStream = new ObjectInputStream(new FileInputStream(filePath));
            TransientTest readObject = (TransientTest)inStream.readObject();
            System.out.println("序列化后:"+readObject.toString());
        } catch (IOException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        } catch (ClassNotFoundException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
    }

3.在main方法中執行,然后可以看到得到的結果為

 

 

印證了上面所講的“用transient關鍵字標記的成員變量不參與序列化過程”。

用二進制查看器打開這個文件也可以看到,數據中只有name1,沒有name2。(請忽略亂碼問題,這個不是重點哈。)

延伸:

在查看JDK源碼的時候會發現很多地方都會加上transient關鍵字來修飾一些屬性,那究竟是出於什么考慮才這么做呢?

我覺得,應該是為了節約磁盤空間,避免造成不必要的浪費吧。

以ArrayList中的 transient Object[] elementData 為例,這個成員變量的注釋為:

 

翻譯出來就是:

/ * *

*存儲ArrayList元素的數組緩沖區。

* ArrayList的容量是這個數組緩沖區的長度。任何

*帶有elementData的空ArrayList == DEFAULTCAPACITY_EMPTY_ELEMENTDATA

*當添加第一個元素時,將被擴展到DEFAULT_CAPACITY。

* /

        這個緩沖區的容量實際上並不是ArrayList的容量,因為其實際上會預留一些空間,當空間不足時還會擴容,為減少浪費,因此在序列化時不會按照默認算法將這個成員變量寫入磁盤。而是寫了個writeObject方法,序列化時會調用這個方法將其持久化,在反序列化是,調用readObject,將其恢復出來。

這2個方法為:

 

 

參考ArrayList,在上面的TransientTest中添加2個方法,見代碼:

public static class TransientTest implements Serializable{
        private static final long serialVersionUID = 233858934995755239L;
        private String name1;
        private transient String name2;
        
        public TransientTest(String name1,String name2){
            this.name1 = name1;
            this.name2 = name2;
        }
        public String toString(){
            return String.format("TransientTest.toString(): name1=%s,name2=%s", name1,name2);
        }
 
        private void writeObject(java.io.ObjectOutputStream s) throws java.io.IOException {
            s.defaultWriteObject();
            s.writeObject(name2);
        }
        private void readObject(java.io.ObjectInputStream s) throws java.io.IOException, ClassNotFoundException {
            s.defaultReadObject();
            name2=String.valueOf(s.readObject());
        }
    }

然后在main方法中執行testTransient(),此時得到的結果是:

 @Transient 注解

 

實體類中使用了@Table注解后,想要添加表中不存在的字段,就要使用@Transient這個注解了。

 

使用 @Transient 表示該屬性並非是一個要映射到數據庫表中的字段,只是起輔助作用.ORM框架將會忽略該屬性

 

注解可以加在屬性上 也可以加在get()上面

 

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM