Java transient 關(guān)鍵字是干啥的
百度百科的解釋:
Java語(yǔ)言的關(guān)鍵字,變量修飾符,如果用transient聲明一個(gè)實(shí)例變量,當(dāng)對(duì)象存儲(chǔ)時(shí),它的值不需要維持。換句話來(lái)說(shuō)就是,用transient關(guān)鍵字標(biāo)記的成員變量不參與序列化過(guò)程。
作用:
Java的serialization提供了一種持久化對(duì)象實(shí)例的機(jī)制。當(dāng)持久化對(duì)象時(shí),可能有一個(gè)特殊的對(duì)象數(shù)據(jù)成員,我們不想用serialization機(jī)制來(lái)保存它。為了在一個(gè)特定對(duì)象的一個(gè)域上關(guān)閉serialization,可以在這個(gè)域前加上關(guān)鍵字transient。當(dāng)一個(gè)對(duì)象被序列化的時(shí)候,transient型變量的值不包括在序列化的表示中,然而非transient型的變量是被包括進(jìn)去的。
編碼試驗(yàn)加以證明:
1.自定義類(為了方便,我直接在main方法所在類中添加的一個(gè)靜態(tài)屬性類,或者叫成員類)
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.寫一個(gè)測(cè)試方法
public static void testTransient(){ String name1="常規(guī)屬性",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方法中執(zhí)行,然后可以看到得到的結(jié)果為
印證了上面所講的“用transient關(guān)鍵字標(biāo)記的成員變量不參與序列化過(guò)程”。
用二進(jìn)制查看器打開這個(gè)文件也可以看到,數(shù)據(jù)中只有name1,沒有name2。(請(qǐng)忽略亂碼問題,這個(gè)不是重點(diǎn)哈。)
延伸:
在查看JDK源碼的時(shí)候會(huì)發(fā)現(xiàn)很多地方都會(huì)加上transient關(guān)鍵字來(lái)修飾一些屬性,那究竟是出于什么考慮才這么做呢?
我覺得,應(yīng)該是為了節(jié)約磁盤空間,避免造成不必要的浪費(fèi)吧。
以ArrayList中的 transient Object[] elementData 為例,這個(gè)成員變量的注釋為:
翻譯出來(lái)就是:
/ * * *存儲(chǔ)ArrayList元素的數(shù)組緩沖區(qū)。 * ArrayList的容量是這個(gè)數(shù)組緩沖區(qū)的長(zhǎng)度。任何 *帶有elementData的空ArrayList == DEFAULTCAPACITY_EMPTY_ELEMENTDATA *當(dāng)添加第一個(gè)元素時(shí),將被擴(kuò)展到DEFAULT_CAPACITY。 * /
這個(gè)緩沖區(qū)的容量實(shí)際上并不是ArrayList的容量,因?yàn)槠鋵?shí)際上會(huì)預(yù)留一些空間,當(dāng)空間不足時(shí)還會(huì)擴(kuò)容,為減少浪費(fèi),因此在序列化時(shí)不會(huì)按照默認(rèn)算法將這個(gè)成員變量寫入磁盤。而是寫了個(gè)writeObject方法,序列化時(shí)會(huì)調(diào)用這個(gè)方法將其持久化,在反序列化是,調(diào)用readObject,將其恢復(fù)出來(lái)。
這2個(gè)方法為:
參考ArrayList,在上面的TransientTest中添加2個(gè)方法,見代碼:
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方法中執(zhí)行testTransient(),此時(shí)得到的結(jié)果是:
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
java中 IO 常用IO操作類繼承結(jié)構(gòu)分析
本篇文章小編為大家介紹,java中 IO 常用IO操作類繼承結(jié)構(gòu)分析。需要的朋友參考下2013-04-04SpringBoot實(shí)現(xiàn)異步事件Event詳解
這篇文章主要介紹了SpringBoot實(shí)現(xiàn)異步事件Event詳解,異步事件的模式,通常將一些非主要的業(yè)務(wù)放在監(jiān)聽器中執(zhí)行,因?yàn)楸O(jiān)聽器中存在失敗的風(fēng)險(xiǎn),所以使用的時(shí)候需要注意,需要的朋友可以參考下2023-11-11淺談Java中Collections.sort對(duì)List排序的兩種方法
本文介紹了Java中Collections.sort對(duì)List排序的兩種方法以及Comparable 與Comparator區(qū)別,文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-12-12Springboot集成Kafka實(shí)現(xiàn)producer和consumer的示例代碼
這篇文章主要介紹了Springboot集成Kafka實(shí)現(xiàn)producer和consumer的示例代碼,詳細(xì)的介紹了什么是Kafka和安裝Kafka以及在springboot項(xiàng)目中集成kafka收發(fā)message,感興趣的小伙伴們可以參考一下2018-05-05