Java小白系列(一):關(guān)鍵字transient

宗旨

本系列意在用最淺顯的文字和代碼示例柑爸,讓大家實實在在的掌握J(rèn)ava最基礎(chǔ)的知識让腹。

一肾砂、前言

我們在學(xué)習(xí)JDK源碼時汛兜,如果大家有留心,應(yīng)該會經(jīng)惩ń瘢看到關(guān)鍵字:transient !但不知道有多少人去真正了解過該關(guān)鍵字的作用肛根。

二辫塌、初識transient

transient,中文意思:短暫的派哲!那么臼氨,它有什么用呢?這就涉及到 Java 持久化機(jī)制芭届。

2.1储矩、Java 持久化

Java為我們提供了一種非常方便的持久化實例機(jī)制:Serializable !

可能有人要說:Serializable 不是用于序列化與反序列化么褂乍?
嗯持隧,對!沒錯逃片,那我想問:一個實例序列化后干啥用屡拨?

一個實例的序列化,可用于數(shù)據(jù)傳輸,然而呀狼,其最大的作用還是用于寫入磁盤裂允,從而防止數(shù)據(jù)丟失。

2.2毙替、Java 序列化

我們在編寫可用于序列化/反序列化的類時抱既,都會去實現(xiàn) Serializable框弛,但是有時候,我們又不想序列化某些字段十饥,那么,我們就可以用關(guān)鍵字 transient 來修飾該字段哩俭。

// BankCard.java
import java.io.Serializable;

public class BankCard implements Serializable {
    private String cardNo;
    private transient String password;

    public String getCardNo() {
        return cardNo;
    }

    public void setCardNo(String cardNo) {
        this.cardNo = cardNo;
    }

    public String getPassword() {
        return password;
    }

    public void setPassword(String password) {
        this.password = password;
    }

    @Override
    public String toString() {
        return "BankCard{cardNo='" + cardNo + '\'' + ", password='" + password + '\'' + '}';
    }
}

編寫測試類

public class Main {
    public static void main(String[] args) {
        try {
            serializeBankCard();
            Thread.sleep(2000);
            deSerializeBankCard();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    private static void serializeBankCard() throws Exception {
        BankCard bankCard = new BankCard();
        bankCard.setCardNo("CardNo: 1234567890");
        bankCard.setPassword("********");

        ObjectOutputStream os = new ObjectOutputStream(new FileOutputStream("./bank.txt"));
        os.writeObject(bankCard);
        os.close();
        System.out.println("序列化 " + bankCard.toString());
    }

    private static void deSerializeBankCard() throws Exception {
        ObjectInputStream is = new ObjectInputStream(new FileInputStream("./bank.txt"));
        BankCard bankCard = (BankCard) is.readObject();
        is.close();
        System.out.println("反序列化 " + bankCard.toString());
    }
}

// 打印輸出
// 序列化 BankCard{cardNo='CardNo: 1234567890', password='********'}
// 反序列化 BankCard{cardNo='CardNo: 1234567890', password='null'}

我們可以看到绷跑,我們先創(chuàng)建『銀行卡』類,然后將數(shù)據(jù)序列化寫到磁盤上凡资,過了2秒后砸捏,我們再從磁盤上讀取數(shù)據(jù),打印發(fā)現(xiàn)隙赁,『password』字段是『null』垦藏。

我們打開磁盤上的文件,用十六進(jìn)制查看數(shù)據(jù)伞访,也可以看到掂骏,文件中,只有『CardNo』厚掷,沒有『password』:

serializable + transient.png

三弟灼、深入分析 transient

在深入分析之前,我先拋出幾個問題冒黑,然后田绑,再帶大家一一去解惑:

  • transient 實現(xiàn)原理;
  • transient 修飾的字段真的無法被序列化抡爹?
  • 靜態(tài)變量能被序列化么掩驱?
  • transient + static 修飾的字段能被序列化么?

OK冬竟,帶著以上問題欧穴,我們?nèi)タ丛创a!

3.1泵殴、Serializable 類

/**
 * ......
 * The writeObject method is responsible for writing the state of the
 * object for its particular class so that the corresponding
 * readObject method can restore it.  The default mechanism for saving
 * the Object's fields can be invoked by calling
 * out.defaultWriteObject. The method does not need to concern
 * itself with the state belonging to its superclasses or subclasses.
 * State is saved by writing the individual fields to the
 * ObjectOutputStream using the writeObject method or by using the
* methods for primitive data types supported by DataOutput.
*
* The readObject method is responsible for reading from the stream and
* restoring the classes fields. It may call in.defaultReadObject to invoke
* the default mechanism for restoring the object's non-static and
* non-transient fields.
* 
* .........
*
* @see java.io.ObjectOutputStream
* @see java.io.ObjectInputStream
* @see java.io.ObjectOutput
* @see java.io.ObjectInput
* @see java.io.Externalizable
* @since   JDK1.1
*/
public interface Serializable {
}

Serializable 的注釋就說到了:

  • writeObject 負(fù)責(zé)寫數(shù)據(jù)涮帘,readObject 負(fù)責(zé)讀數(shù)據(jù);
  • 數(shù)據(jù)(狀態(tài))的讀寫笑诅,并不關(guān)心它的父類或子類焚辅;
  • ObjectOutputSteam 是基于 DataOutput 實現(xiàn)的寫映屋;
  • readObject 負(fù)責(zé)讀流(stream),并存到類成員變量中同蜻,它的默認(rèn)存到非static成員和非transient成員中棚点;

3.2、ObjectOutputStream 類

開頭的注釋也說了:默認(rèn)的 Serializable 機(jī)制湾蔓,只寫類的對象瘫析、簽名以及非 transient 和非 static 字段

/*
 * The default serialization mechanism for an object writes the class of the
 * object, the class signature, and the values of all non-transient and
 * non-static fields.
 */
 public class ObjectOutputStream extends OutputStream implements ObjectOutput, ObjectStreamConstants {
    private final BlockDataOutputStream bout;
    private static class BlockDataOutputStream extends OutputStream implements DataOutput {
        ......
    }
    
    public final void writeObject(Object obj) throws IOException {
        ......
        try {
            writeObject0(obj, false);
        }
        ......
    }
    
    private void writeObject0(Object obj, boolean unshared) throws IOException {
        boolean oldMode = bout.setBlockDataMode(false);
        depth++;
        try {
            ......
            
            ObjectStreamClass desc;
            for (;;) {
                ......
                // 重點(diǎn):獲取 Serializable對象的 desc
                // desc 中的 fields 會過濾 static 和 transient
                // 具體代碼流程繼續(xù)看 3.3
                desc = ObjectStreamClass.lookup(cl, true);
                ......
            }
            

            } else if (obj instanceof Serializable) {
                writeOrdinaryObject(obj, desc, unshared); // obj 根據(jù) desc.fields 序列化數(shù)據(jù)
            }
            ......
            
        } finally {
            depth--;
            bout.setBlockDataMode(oldMode);
        }
    }
    
    private void writeOrdinaryObject(Object obj,
                                     ObjectStreamClass desc,
                                     boolean unshared) throws IOException {
        if (extendedDebugInfo) {
            debugInfoStack.push(
                (depth == 1 ? "root " : "") + "object (class \"" +
                obj.getClass().getName() + "\", " + obj.toString() + ")");
        }
        try {
            desc.checkSerialize();
            // 最終調(diào)用 bout 來寫數(shù)據(jù)
            bout.writeByte(TC_OBJECT);
            
            // 所有的 writeXXX 都將調(diào)用 bout 來寫數(shù)據(jù)
            writeClassDesc(desc, false);
            handles.assign(unshared ? null : obj);
            if (desc.isExternalizable() && !desc.isProxy()) {
                writeExternalData((Externalizable) obj);
            } else {
                writeSerialData(obj, desc);
            }
        } finally {
            if (extendedDebugInfo) {
                debugInfoStack.pop();
            }
        }
    }
}

3.3、ObjectStreamClass類

3.3.1默责、ObjectStreamClass.lookup

public class ObjectStreamClass implements Serializable {
    static ObjectStreamClass lookup(Class<?> cl, boolean all) {
        ......
        if (entry == null) {
            try {
                entry = new ObjectStreamClass(cl);
            } catch (Throwable th) {
                entry = th;
            }
           ......
        }
    
        ......
    }
}

3.3.2贬循、ObjectStreamClass構(gòu)造函數(shù)

public class ObjectStreamClass implements Serializable {
    private ObjectStreamClass(final Class<?> cl) {
        ......
        if (serializable) {
            AccessController.doPrivileged(new PrivilegedAction<Void>() {
                public Void run() {
                    ......
                    try {
                        fields = getSerialFields(cl);  // 獲取可序列化的字段
                        computeFieldOffsets();
                    } catch (InvalidClassException e) {
                        ......
                        fields = NO_FIELDS;
                    }
                    ......
                }
            }
        }
    }
}

3.3.3、ObjectStreamClass.getSerialFields

public class ObjectStreamClass implements Serializable {
    private static ObjectStreamField[] getSerialFields(Class<?> cl) throws InvalidClassException {
        ObjectStreamField[] fields;
        if (Serializable.class.isAssignableFrom(cl) &&
            !Externalizable.class.isAssignableFrom(cl) &&
            !Proxy.isProxyClass(cl) &&
            !cl.isInterface())
        {
            if ((fields = getDeclaredSerialFields(cl)) == null) {
                fields = getDefaultSerialFields(cl);
            }
            Arrays.sort(fields);
        } else {
            fields = NO_FIELDS;
        }
        return fields;
    }
}

3.3.4桃序、ObjectStreamClass.getDefaultSerialFields

public class ObjectStreamClass implements Serializable {
    private static ObjectStreamField[] getDefaultSerialFields(Class<?> cl) {
        Field[] clFields = cl.getDeclaredFields();
        ArrayList<ObjectStreamField> list = new ArrayList<>();
        int mask = Modifier.STATIC | Modifier.TRANSIENT; // 重點(diǎn)1:掩碼 mask
    
        for (int i = 0; i < clFields.length; i++) {
            // 重點(diǎn)2:過濾 static 或者 transient
            if ((clFields[i].getModifiers() & mask) == 0) {
                list.add(new ObjectStreamField(clFields[i], false, true));
            }
        }
        int size = list.size();
        return (size == 0) ? NO_FIELDS : list.toArray(new ObjectStreamField[size]);
    }
}

3.4杖虾、斷點(diǎn)查看運(yùn)行時JDK源碼

jdk_runtime.png

簡單的看完以上源碼,至少回答了我們這么幾個問題:

  • static 變量是無法序列化的媒熊,無論是否加了 transient 修飾符奇适;
  • transient 修飾的變量也是無法序列化的;

那么芦鳍,還有一個問題嚷往,transient 修飾的變量真的無法序列化么?

再回答該問題前柠衅,首先我想問下大家皮仁,大家知道 Java 除了提供 Serializable 序列化外,還有提供其它方式么菲宴?

答案是:Java 提供了兩種序列化方式

  1. Serializable贷祈,序列化與反序列化由 Java 實現(xiàn)(ObjectOutputStream / ObjectInputStream 的默認(rèn)機(jī)制);
  2. Externalizable喝峦,序列化與反序列化由開發(fā)者自主來實現(xiàn)势誊;

先來看看示例

// BankCardExt.java
import java.io.Externalizable;
import java.io.IOException;
import java.io.ObjectInput;
import java.io.ObjectOutput;

public class BankCardExt implements Externalizable {
    private String cardNo;
    private transient String password;

    public String getCardNo() {
        return cardNo;
    }

    public void setCardNo(String cardNo) {
        this.cardNo = cardNo;
    }

    public String getPassword() {
        return password;
    }

    public void setPassword(String password) {
        this.password = password;
    }

    @Override
    public void writeExternal(ObjectOutput out) throws IOException {
        out.writeObject(cardNo);
        out.writeObject(password);
    }

    @Override
    public void readExternal(ObjectInput in) throws IOException, ClassNotFoundException {
        cardNo = (String) in.readObject();
        password = (String) in.readObject();
    }

    @Override
    public String toString() {
        return "BankCardExt{cardNo='" + cardNo + "\', password='" + password + "\'}";
    }
}

這個類的讀和寫,由我們來實現(xiàn)愈犹。

再來個測試類

public class Main {

    public static void main(String[] args) {
        try {
            serializeBankCard();
            Thread.sleep(2000);
            deSerializeBankCard();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    private static void serializeBankCard() throws Exception {
        BankCardExt bankCard = new BankCardExt();
        bankCard.setCardNo("CardNo: 1234567890");
        bankCard.setPassword("********");

        ObjectOutputStream os = new ObjectOutputStream(new FileOutputStream("./bankExt.txt"));
        os.writeObject(bankCard);
        os.close();
        System.out.println("序列化 " + bankCard.toString());
    }

    private static void deSerializeBankCard() throws Exception {
        ObjectInputStream is = new ObjectInputStream(new FileInputStream("./bankExt.txt"));
        BankCardExt bankCard = (BankCardExt) is.readObject();
        is.close();
        System.out.println("反序列化 " + bankCard.toString());
    }
}

// 打印輸出
// 序列化 BankCardExt{cardNo='CardNo: 1234567890', password='********'}
// 反序列化 BankCardExt{cardNo='CardNo: 1234567890', password='********'}

我們看到,讀的時候闻丑,數(shù)據(jù)也回來了漩怎。

exteralizable.png

查看文件,我們也看到了『CardNo』和『password』嗦嗡。

這里還有一點(diǎn)需要注意:寫的順序與讀的順序要保持一致勋锤!如果我們順充不一致,結(jié)果將如下:

{
    @Override
    public void writeExternal(ObjectOutput out) throws IOException {
        out.writeObject(cardNo);
        out.writeObject(password);
    }
    
    @Override
    public void readExternal(ObjectInput in) throws IOException, ClassNotFoundException {
        password = (String) in.readObject();
        cardNo = (String) in.readObject();
    }
}

// 打印輸出
// 序列化 BankCardExt{cardNo='CardNo: 1234567890', password='********'}
// 反序列化 BankCardExt{cardNo='********', password='CardNo: 1234567890'}

OK侥祭,最后一個問題我們也解答了叁执,當(dāng)我們實現(xiàn) Externalizable 時茄厘,需要我們自主選擇哪些字段需要序列化與反序列化,因此谈宛,此時的 transient 修飾符將沒有作用次哈。

四、總結(jié)

本文分析了 transient 的序列化底層機(jī)制:

  • 當(dāng)我們的類實現(xiàn)了 Serializable 時吆录,transient 修飾的變量只能存在于內(nèi)存中窑滞,ObjectOutputStream 將忽略 transient 和 static 修飾的變量;
  • 當(dāng)我們的類實現(xiàn)了 Externalizable 時恢筝,此時 transient 關(guān)鍵字將失效哀卫,類的成員變量的序列化與反序列化將由我們自己控制;
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末撬槽,一起剝皮案震驚了整個濱河市此改,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌侄柔,老刑警劉巖共啃,帶你破解...
    沈念sama閱讀 216,744評論 6 502
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異勋拟,居然都是意外死亡勋磕,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,505評論 3 392
  • 文/潘曉璐 我一進(jìn)店門敢靡,熙熙樓的掌柜王于貴愁眉苦臉地迎上來挂滓,“玉大人,你說我怎么就攤上這事啸胧「险荆” “怎么了?”我有些...
    開封第一講書人閱讀 163,105評論 0 353
  • 文/不壞的土叔 我叫張陵纺念,是天一觀的道長贝椿。 經(jīng)常有香客問我,道長陷谱,這世上最難降的妖魔是什么烙博? 我笑而不...
    開封第一講書人閱讀 58,242評論 1 292
  • 正文 為了忘掉前任,我火速辦了婚禮烟逊,結(jié)果婚禮上渣窜,老公的妹妹穿的比我還像新娘。我一直安慰自己宪躯,他們只是感情好乔宿,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,269評論 6 389
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著访雪,像睡著了一般详瑞。 火紅的嫁衣襯著肌膚如雪掂林。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,215評論 1 299
  • 那天坝橡,我揣著相機(jī)與錄音泻帮,去河邊找鬼。 笑死驳庭,一個胖子當(dāng)著我的面吹牛刑顺,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播饲常,決...
    沈念sama閱讀 40,096評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼蹲堂,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了贝淤?” 一聲冷哼從身側(cè)響起柒竞,我...
    開封第一講書人閱讀 38,939評論 0 274
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎播聪,沒想到半個月后朽基,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,354評論 1 311
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡离陶,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,573評論 2 333
  • 正文 我和宋清朗相戀三年稼虎,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片招刨。...
    茶點(diǎn)故事閱讀 39,745評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡霎俩,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出沉眶,到底是詐尸還是另有隱情打却,我是刑警寧澤,帶...
    沈念sama閱讀 35,448評論 5 344
  • 正文 年R本政府宣布谎倔,位于F島的核電站柳击,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏片习。R本人自食惡果不足惜捌肴,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,048評論 3 327
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望藕咏。 院中可真熱鬧状知,春花似錦、人聲如沸侈离。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,683評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽卦碾。三九已至铺坞,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間洲胖,已是汗流浹背济榨。 一陣腳步聲響...
    開封第一講書人閱讀 32,838評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留绿映,地道東北人擒滑。 一個月前我還...
    沈念sama閱讀 47,776評論 2 369
  • 正文 我出身青樓,卻偏偏與公主長得像叉弦,于是被迫代替她去往敵國和親丐一。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,652評論 2 354

推薦閱讀更多精彩內(nèi)容