摘要:由以上結(jié)果分析可知,靜態(tài)變量不能被序列化,示例讀取出來的是在內(nèi)存中存儲的值。關(guān)鍵字總結(jié)修飾的變量不能被序列化只作用于實現(xiàn)接口只能用來修飾普通成員變量字段不管有沒有修飾,靜態(tài)變量都不能被序列化好了,棧長花了半天時間,終于整理完了。
先解釋下什么是序列化
我們的對象并不只是存在內(nèi)存中,還需要傳輸網(wǎng)絡(luò),或者保存起來下次再加載出來用,所以需要Java序列化技術(shù)。
Java序列化技術(shù)正是將對象轉(zhuǎn)變成一串由二進制字節(jié)組成的數(shù)組,可以通過將二進制數(shù)據(jù)保存到磁盤或者傳輸網(wǎng)絡(luò),磁盤或者網(wǎng)絡(luò)接收者可以在對象的屬類的模板上來反序列化類的對象,達(dá)到對象持久化的目的。
更多序列化請參考:《關(guān)于Java序列化你應(yīng)該知道的一切》這篇文章。
什么是 transient?簡單來說就是,被 transient 修飾的變量不能被序列化。
具體來看下面的示例1
import java.io.FileInputStream; import java.io.FileOutputStream; import java.io.ObjectInputStream; import java.io.ObjectOutputStream; import java.io.Serializable; /** * @author 微信公眾號:Java技術(shù)棧 */ public class TransientTest { public static void main(String[] args) throws Exception { User user = new User(); user.setUsername("Java技術(shù)棧"); user.setId("javastack"); System.out.println(" 序列化之前"); System.out.println("username: " + user.getUsername()); System.out.println("id: " + user.getId()); ObjectOutputStream os = new ObjectOutputStream(new FileOutputStream("d:/user.txt")); os.writeObject(user); os.flush(); os.close(); ObjectInputStream is = new ObjectInputStream(new FileInputStream("d:/user.txt")); user = (User) is.readObject(); is.close(); System.out.println(" 序列化之后"); System.out.println("username: " + user.getUsername()); System.out.println("id: " + user.getId()); } } /** * @author 微信公眾號:Java技術(shù)棧 */ class User implements Serializable { private static final long serialVersionUID = 1L; private String username; private transient String id; public String getUsername() { return username; } public void setUsername(String username) { this.username = username; } public String getId() { return id; } public void setId(String id) { this.id = id; } }
輸出結(jié)果:
序列化之前 username: Java技術(shù)棧 id: javastack 序列化之后 username: Java技術(shù)棧 id: null
示例1在 id 字段上加了 transient 關(guān)鍵字修飾,反序列化出來之后值為 null,說明了被 transient 修飾的變量不能被序列化。
靜態(tài)變量能被序列化嗎?這個話題也是最近棧長的Java技術(shù)棧vip群里面討論的,大家對這個知識點比較模糊,我就寫了這篇文章測試總結(jié)一下。
如果你也想加入我們的Java技術(shù)棧vip群和各位大牛一起討論技術(shù),那點擊這個鏈接了解加入吧。
那么,到底靜態(tài)變量能被序列化嗎?廢話少說,先動手測試下吧!
示例2:
import java.io.FileInputStream; import java.io.FileOutputStream; import java.io.ObjectInputStream; import java.io.ObjectOutputStream; import java.io.Serializable; /** * @author 微信公眾號:Java技術(shù)棧 */ public class TransientStaticTest { public static void main(String[] args) throws Exception { User2 user = new User2(); User2.username = "Java技術(shù)棧1"; user.setId("javastack"); System.out.println(" 序列化之前"); System.out.println("username: " + user.getUsername()); System.out.println("id: " + user.getId()); ObjectOutputStream os = new ObjectOutputStream(new FileOutputStream("d:/user.txt")); os.writeObject(user); os.flush(); os.close(); // 在反序列化出來之前,改變靜態(tài)變量的值 User2.username = "Java技術(shù)棧2"; ObjectInputStream is = new ObjectInputStream(new FileInputStream("d:/user.txt")); user = (User2) is.readObject(); is.close(); System.out.println(" 序列化之后"); System.out.println("username: " + user.getUsername()); System.out.println("id: " + user.getId()); } } /** * @author 微信公眾號:Java技術(shù)棧 */ class User2 implements Serializable { private static final long serialVersionUID = 1L; public static String username; private transient String id; public String getUsername() { return username; } public String getId() { return id; } public void setId(String id) { this.id = id; } }
輸出結(jié)果:
序列化之前 username: Java技術(shù)棧1 id: javastack 序列化之后 username: Java技術(shù)棧2 id: null
示例2把 username 改為了 public static, 并在反序列化出來之前改變了靜態(tài)變量的值,結(jié)果可以看出序列化之后的值并非序列化進去時的值。
由以上結(jié)果分析可知,靜態(tài)變量不能被序列化,示例2讀取出來的是 username 在 JVM 內(nèi)存中存儲的值。
transient 真不能被序列化嗎?繼續(xù)來看示例3:
import java.io.Externalizable; import java.io.File; import java.io.FileInputStream; import java.io.FileOutputStream; import java.io.IOException; import java.io.ObjectInput; import java.io.ObjectInputStream; import java.io.ObjectOutput; import java.io.ObjectOutputStream; /** * @author 微信公眾號:Java技術(shù)棧 */ public class ExternalizableTest { public static void main(String[] args) throws Exception { User3 user = new User3(); user.setUsername("Java技術(shù)棧"); user.setId("javastack"); ObjectOutput objectOutput = new ObjectOutputStream(new FileOutputStream(new File("javastack"))); objectOutput.writeObject(user); ObjectInput objectInput = new ObjectInputStream(new FileInputStream(new File("javastack"))); user = (User3) objectInput.readObject(); System.out.println(user.getUsername()); System.out.println(user.getId()); objectOutput.close(); objectInput.close(); } } /** * @author 微信公眾號:Java技術(shù)棧 */ class User3 implements Externalizable { private static final long serialVersionUID = 1L; public User3() { } private String username; private transient String id; public String getUsername() { return username; } public void setUsername(String username) { this.username = username; } public String getId() { return id; } public void setId(String id) { this.id = id; } @Override public void writeExternal(ObjectOutput objectOutput) throws IOException { objectOutput.writeObject(id); } @Override public void readExternal(ObjectInput objectInput) throws IOException, ClassNotFoundException { id = (String) objectInput.readObject(); } }
輸出結(jié)果:
null javastack
示例3的 id 被 transient 修改了,為什么還能序列化出來?那是因為 User3 實現(xiàn)了接口 Externalizable,而不是 Serializable。
在 Java 中有兩種實現(xiàn)序列化的方式,Serializable 和 Externalizable,可能大部分人只知道 Serializable 而不知道 Externalizable。
這兩種序列化方式的區(qū)別是:實現(xiàn)了 Serializable 接口是自動序列化的,實現(xiàn) Externalizable 則需要手動序列化,通過 writeExternal 和 readExternal 方法手動進行,這也是為什么上面的 username 為 null 的原因了。
transient 關(guān)鍵字總結(jié)1)transient修飾的變量不能被序列化;
2)transient只作用于實現(xiàn) Serializable 接口;
3)transient只能用來修飾普通成員變量字段;
4)不管有沒有 transient 修飾,靜態(tài)變量都不能被序列化;
好了,棧長花了半天時間,終于整理完了。如果對你有幫助,那就轉(zhuǎn)發(fā)分享一下吧!如果你也想加入我們的Java技術(shù)棧vip群和各位大牛一起討論技術(shù),那點擊這個鏈接了解加入吧
另外,棧長已經(jīng)整理了大量 Java 系列核心技術(shù)知識點文章,關(guān)注Java技術(shù)棧微信公眾號,在后臺回復(fù)關(guān)鍵字:java,即可獲取最新版。
本文原創(chuàng)首發(fā)于微信公眾號:Java技術(shù)棧(id:javastack),關(guān)注公眾號在后臺回復(fù) "java" 可獲取更多,轉(zhuǎn)載請原樣保留本信息。
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/73573.html
摘要:為什么叫重入鎖呢,我們把它拆開來看就明了了。釋放鎖,每次鎖持有者數(shù)量遞減,直到為止。 相信大家在工作或者面試過程中經(jīng)常聽到重入鎖這個概念,或者與關(guān)鍵字 synchrozied 的對比,棧長面試了這么多人,80%的面試者都沒有答對或沒有答到點上,或者把雙重效驗鎖搞混了,哭笑不得。。 那么你對重入鎖了解有多少呢?今天,棧長幫大家撕開重入鎖的面紗,來見識下重入鎖的真實容顏。。 什么是重入鎖 ...
摘要:記得幾年前有一次棧長去面試,問到了這么一個問題中的對象都是在堆中分配嗎說明為什么當(dāng)時我被問得一臉蒙逼,瞬間被秒殺得體無完膚,當(dāng)時我壓根就不知道他在考什么知識點,難道對象不是在堆中分配嗎最后就沒然后了,回去等通知了。。 記得幾年前有一次棧長去面試,問到了這么一個問題: Java中的對象都是在堆中分配嗎?說明為什么! 當(dāng)時我被問得一臉蒙逼,瞬間被秒殺得體無完膚,當(dāng)時我壓根就不知道他在考什么...
摘要:加載因子是哈希表在其容量自動增加之前可以達(dá)到多滿的一種尺度。當(dāng)哈希表中的條目數(shù)超出了加載因子與當(dāng)前容量的乘積時,則要對該哈希表進行操作即重建內(nèi)部數(shù)據(jù)結(jié)構(gòu),從而哈希表將具有大約兩倍的桶數(shù)。 showImg(https://upload-images.jianshu.io/upload_images/4565148-98b22ba5ae7d9723.jpg?imageMogr2/auto-...
摘要:曾幾何時,閉包好像就是一個十分難以捉摸透的東西,看了很多文章,對閉包都各有說法,以致讓我十分暈,什么內(nèi)部變量外部變量的,而且大多數(shù)都只描述一個過程,沒有給閉包的定義,最后,舉幾個例子,告訴你這就是閉包。 曾幾何時,閉包好像就是一個十分難以捉摸透的東西,看了很多文章,對閉包都各有說法,以致讓我十分暈,什么內(nèi)部變量、外部變量的,而且大多數(shù)都只描述一個過程,沒有給閉包的定義,最后,舉幾個例子...
閱讀 723·2021-10-14 09:42
閱讀 1976·2021-09-22 15:04
閱讀 1585·2019-08-30 12:44
閱讀 2146·2019-08-29 13:29
閱讀 2738·2019-08-29 12:51
閱讀 556·2019-08-26 18:18
閱讀 707·2019-08-26 13:43
閱讀 2818·2019-08-26 13:38