成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

小伙子,你真的搞懂 transient 關(guān)鍵字了嗎?

curlyCheng / 3050人閱讀

摘要:由以上結(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

相關(guān)文章

  • 到底什么是重入鎖,拜托,一次搞清楚!

    摘要:為什么叫重入鎖呢,我們把它拆開來看就明了了。釋放鎖,每次鎖持有者數(shù)量遞減,直到為止。 相信大家在工作或者面試過程中經(jīng)常聽到重入鎖這個概念,或者與關(guān)鍵字 synchrozied 的對比,棧長面試了這么多人,80%的面試者都沒有答對或沒有答到點上,或者把雙重效驗鎖搞混了,哭笑不得。。 那么你對重入鎖了解有多少呢?今天,棧長幫大家撕開重入鎖的面紗,來見識下重入鎖的真實容顏。。 什么是重入鎖 ...

    LiuRhoRamen 評論0 收藏0
  • 面試問我 Java 逃逸分析,瞬間被秒殺了。。

    摘要:記得幾年前有一次棧長去面試,問到了這么一個問題中的對象都是在堆中分配嗎說明為什么當(dāng)時我被問得一臉蒙逼,瞬間被秒殺得體無完膚,當(dāng)時我壓根就不知道他在考什么知識點,難道對象不是在堆中分配嗎最后就沒然后了,回去等通知了。。 記得幾年前有一次棧長去面試,問到了這么一個問題: Java中的對象都是在堆中分配嗎?說明為什么! 當(dāng)時我被問得一臉蒙逼,瞬間被秒殺得體無完膚,當(dāng)時我壓根就不知道他在考什么...

    CastlePeaK 評論0 收藏0
  • HashMap 真的了解嗎?

    摘要:加載因子是哈希表在其容量自動增加之前可以達(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-...

    RdouTyping 評論0 收藏0
  • 作用域閉包,真的了嗎?

    摘要:曾幾何時,閉包好像就是一個十分難以捉摸透的東西,看了很多文章,對閉包都各有說法,以致讓我十分暈,什么內(nèi)部變量外部變量的,而且大多數(shù)都只描述一個過程,沒有給閉包的定義,最后,舉幾個例子,告訴你這就是閉包。 曾幾何時,閉包好像就是一個十分難以捉摸透的東西,看了很多文章,對閉包都各有說法,以致讓我十分暈,什么內(nèi)部變量、外部變量的,而且大多數(shù)都只描述一個過程,沒有給閉包的定義,最后,舉幾個例子...

    yangrd 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<