一個(gè)Intent與LinkedHashMap的小問(wèn)題

前言

這周QA報(bào)了一個(gè)小bug,頁(yè)面A傳給頁(yè)面B的數(shù)據(jù)順序不對(duì),查了一下代碼,原來(lái)頁(yè)面A中數(shù)據(jù)存儲(chǔ)容器用的是HashMap,而HasMap存取是無(wú)序的,所以傳給B去讀數(shù)據(jù)的時(shí)候,自然順序不對(duì)。

解決

既然HashMap是無(wú)序的,那我直接用LinkedHashMap來(lái)代替不就行了,大多數(shù)人估計(jì)看到這個(gè)bug時(shí),開(kāi)始都是這么想的。于是我就順手在HashMap前加了一個(gè)Linked,點(diǎn)了一下run,泯上一口茶,靜靜等待著奇跡的發(fā)生。

然而奇跡沒(méi)有來(lái)臨,奇怪的事反倒是發(fā)生了,B頁(yè)面收到數(shù)據(jù)后,居然報(bào)了一個(gè)類(lèi)型強(qiáng)轉(zhuǎn)錯(cuò)誤,B收到的是HashMap,而不是LinkedHashMap,怎么可能!!!!我趕緊放下茶杯,review了一下代碼,沒(méi)錯(cuò)啊,A頁(yè)面?zhèn)鬟f的確實(shí)是LinkedHashMap,但是B拿到就是HashMap,真是活見(jiàn)鬼了。

我立馬Google了一下,遇到這個(gè)錯(cuò)誤的人還真不少,評(píng)論區(qū)給出的一種解決方案就是用Gson將LinkedHashMap序列化成String,再進(jìn)行傳遞。。。由于bug催的緊,我也沒(méi)有去嘗試這種方法了,直接就放棄了傳遞Map,改用ArrayList了。不過(guò)后來(lái)看源碼,又發(fā)現(xiàn)了另外一種方式,稍后再說(shuō)。

原因

Bug倒是解決了,但是Intent無(wú)法傳遞LinkedHashMap的問(wèn)題還在我腦海里縈繞,我就稍微翻看了一下源碼,恍然大悟!

HashMap實(shí)現(xiàn)了Serializable接口,而LinkedHashMap是繼承自HashMap的,所以用Intent傳遞是沒(méi)有問(wèn)題的,我們先來(lái)追一下A頁(yè)面?zhèn)鬟f的地方:

intent.putExtra("map",new LinkedHashMap<>());

接著往里看:

public Intent putExtra(String name, Serializable value) {
    if (mExtras == null) {
        mExtras = new Bundle();
    }
    mExtras.putSerializable(name, value);
    return this;
}

intent是直接構(gòu)造了一個(gè)Bundle,將數(shù)據(jù)傳遞到Bundle里,Bundle.putSerializable()里其實(shí)也是直接調(diào)用了父類(lèi)BaseBundle.putSerializable():

void putSerializable(@Nullable String key, @Nullable Serializable value) {
    unparcel();
    mMap.put(key, value);
}

這里直接將value放入了一個(gè)ArrayMap中,并沒(méi)有做什么特殊處理。

事情到這似乎沒(méi)有了下文,那么這個(gè)LinkedHashMap又是何時(shí)轉(zhuǎn)為HashMap的呢?有沒(méi)有可能是在startActivity()中做的處理呢?

了解activity啟動(dòng)流程的工程師應(yīng)該清楚,startActivity()最后調(diào)的是:

 ActivityManagerNative.getDefault().startActivity()

ActivityManagerNative是個(gè)Binder對(duì)象,其功能實(shí)現(xiàn)是在ActivityManagerService中,而其在app進(jìn)程中的代理對(duì)象則為ActivityManagerProxy。所以上面的startActivity()最后調(diào)用的是ActivityManagerProxy.startActivity(),我們來(lái)看看這個(gè)方法的源碼:

public int startActivity(IApplicationThread caller, String callingPackage, Intent intent,
        String resolvedType, IBinder resultTo, String resultWho, int requestCode,
        int startFlags, ProfilerInfo profilerInfo, Bundle options) throws RemoteException {
    Parcel data = Parcel.obtain();
    Parcel reply = Parcel.obtain();
    ......
    intent.writeToParcel(data, 0);
    ......
    int result = reply.readInt();
    reply.recycle();
    data.recycle();
    return result;
}

注意到方法中調(diào)用了intent.writeToParcel(data, 0),難道這里做了什么特殊處理?

public void writeToParcel(Parcel out, int flags) {
    out.writeString(mAction);
    Uri.writeToParcel(out, mData);
    out.writeString(mType);
    out.writeInt(mFlags);
    out.writeString(mPackage);
    ......
    out.writeBundle(mExtras);
}

最后一行調(diào)用了Parcel.writeBundle()方法,傳參為mExtras,而之前的LinkedHashMap就放在這mExtras中。

    public final void writeBundle(Bundle val) {
    if (val == null) {
        writeInt(-1);
        return;
    }

    val.writeToParcel(this, 0);
}

這里最后調(diào)用了Bundle.writeToParcel(),最終會(huì)調(diào)用到其父類(lèi)BaseBundle的writeToParcelInner():

void writeToParcelInner(Parcel parcel, int flags) {
    // Keep implementation in sync with writeToParcel() in
    // frameworks/native/libs/binder/PersistableBundle.cpp.
    final Parcel parcelledData;
    synchronized (this) {
        parcelledData = mParcelledData;
    }
    if (parcelledData != null) {
       ......
    } else {
        // Special case for empty bundles.
        if (mMap == null || mMap.size() <= 0) {
            parcel.writeInt(0);
            return;
        }
        ......
        parcel.writeArrayMapInternal(mMap);
        ......
    }
}

可見(jiàn)最后else分支里,會(huì)調(diào)用Parcel.writeArrayMapInternal(mMap),這個(gè)mMap即為Bundle中存儲(chǔ)K-V的ArrayMap,看看這里有沒(méi)有對(duì)mMap做特殊處理:

void writeArrayMapInternal(ArrayMap<String, Object> val) {
    if (val == null) {
        writeInt(-1);
        return;
    }
    // Keep the format of this Parcel in sync with writeToParcelInner() in
    // frameworks/native/libs/binder/PersistableBundle.cpp.
    final int N = val.size();
    writeInt(N);
    if (DEBUG_ARRAY_MAP) {
        RuntimeException here =  new RuntimeException("here");
        here.fillInStackTrace();
        Log.d(TAG, "Writing " + N + " ArrayMap entries", here);
    }
    int startPos;
    for (int i=0; i<N; i++) {
        if (DEBUG_ARRAY_MAP) startPos = dataPosition();
        writeString(val.keyAt(i));
        writeValue(val.valueAt(i));
        if (DEBUG_ARRAY_MAP) Log.d(TAG, "  Write #" + i + " "
                + (dataPosition()-startPos) + " bytes: key=0x"
                + Integer.toHexString(val.keyAt(i) != null ? val.keyAt(i).hashCode() : 0)
                + " " + val.keyAt(i));
    }
}

在最后的for循環(huán)中,會(huì)遍歷mMap中所有的K-V對(duì),先調(diào)用writeString()寫(xiě)入Key,再調(diào)用writeValue()來(lái)寫(xiě)入Value。真相就在writeValue()里:

public final void writeValue(Object v) {
    if (v == null) {
        writeInt(VAL_NULL);
    } else if (v instanceof String) {
        writeInt(VAL_STRING);
        writeString((String) v);
    } else if (v instanceof Integer) {
        writeInt(VAL_INTEGER);
        writeInt((Integer) v);
    } else if (v instanceof Map) {
        writeInt(VAL_MAP);
        writeMap((Map) v);
    } 
    ......
    ......
}

這里會(huì)判斷value的具體類(lèi)型,如果是Map類(lèi)型,會(huì)先寫(xiě)入一個(gè)VAL_MAP的類(lèi)型常量,緊接著調(diào)用writeMap()寫(xiě)入value。writeMap()最后走到了writeMapInternal():

void writeMapInternal(Map<String,Object> val) {
    if (val == null) {
        writeInt(-1);
        return;
    }
    Set<Map.Entry<String,Object>> entries = val.entrySet();
    writeInt(entries.size());
    for (Map.Entry<String,Object> e : entries) {
        writeValue(e.getKey());
        writeValue(e.getValue());
    }
}

可見(jiàn),這里并沒(méi)有直接將LinkedHashMap序列化,而是遍歷其中所有K-V,依次寫(xiě)入每個(gè)Key和Value,所以L(fǎng)inkedHashMap到這時(shí)就已經(jīng)失去意義了。

那么B頁(yè)面在讀取這個(gè)LinkedHashMap的時(shí)候,是什么情況呢?從Intent中讀取數(shù)據(jù)時(shí),最終會(huì)走到getSerializable():

Serializable getSerializable(@Nullable String key) {
    unparcel();
    Object o = mMap.get(key);
    if (o == null) {
        return null;
    }
    try {
        return (Serializable) o;
    } catch (ClassCastException e) {
        typeWarning(key, o, "Serializable", e);
        return null;
    }
}

這里乍一看就是直接從mMap中通過(guò)key取到value,其實(shí)重要的邏輯全都在第一句unparcel()中:

synchronized void unparcel() {
    synchronized (this) {
        final Parcel parcelledData = mParcelledData;
        if (parcelledData == null) {
            if (DEBUG) Log.d(TAG, "unparcel "
                    + Integer.toHexString(System.identityHashCode(this))
                    + ": no parcelled data");
            return;
        }

        if (LOG_DEFUSABLE && sShouldDefuse && (mFlags & FLAG_DEFUSABLE) == 0) {
            Slog.wtf(TAG, "Attempting to unparcel a Bundle while in transit; this may "
                    + "clobber all data inside!", new Throwable());
        }

        if (isEmptyParcel()) {
            if (DEBUG) Log.d(TAG, "unparcel "
                    + Integer.toHexString(System.identityHashCode(this)) + ": empty");
            if (mMap == null) {
                mMap = new ArrayMap<>(1);
            } else {
                mMap.erase();
            }
            mParcelledData = null;
            return;
        }

        int N = parcelledData.readInt();
        if (DEBUG) Log.d(TAG, "unparcel " + Integer.toHexString(System.identityHashCode(this))
                + ": reading " + N + " maps");
        if (N < 0) {
            return;
        }
        ArrayMap<String, Object> map = mMap;
        if (map == null) {
            map = new ArrayMap<>(N);
        } else {
            map.erase();
            map.ensureCapacity(N);
        }
        try {
            parcelledData.readArrayMapInternal(map, N, mClassLoader);
        } catch (BadParcelableException e) {
            if (sShouldDefuse) {
                Log.w(TAG, "Failed to parse Bundle, but defusing quietly", e);
                map.erase();
            } else {
                throw e;
            }
        } finally {
            mMap = map;
            parcelledData.recycle();
            mParcelledData = null;
        }
        if (DEBUG) Log.d(TAG, "unparcel " + Integer.toHexString(System.identityHashCode(this))
                + " final map: " + mMap);
    }

這里主要是讀取數(shù)據(jù),然后填充到mMap中,其中關(guān)鍵點(diǎn)在于parcelledData.readArrayMapInternal(map, N, mClassLoader):

void readArrayMapInternal(ArrayMap outVal, int N,
    ClassLoader loader) {
    if (DEBUG_ARRAY_MAP) {
        RuntimeException here =  new RuntimeException("here");
        here.fillInStackTrace();
        Log.d(TAG, "Reading " + N + " ArrayMap entries", here);
    }
    int startPos;
    while (N > 0) {
        if (DEBUG_ARRAY_MAP) startPos = dataPosition();
        String key = readString();
        Object value = readValue(loader);
        if (DEBUG_ARRAY_MAP) Log.d(TAG, "  Read #" + (N-1) + " "
                + (dataPosition()-startPos) + " bytes: key=0x"
                + Integer.toHexString((key != null ? key.hashCode() : 0)) + " " + key);
        outVal.append(key, value);
        N--;
    }
    outVal.validate();
}

這里其實(shí)對(duì)應(yīng)于之前所說(shuō)的writeArrayMapInternal(),先調(diào)用readString讀出Key值,再調(diào)用readValue()讀取value值,所以重點(diǎn)還是在于readValue():

public final Object readValue(ClassLoader loader) {
    int type = readInt();

    switch (type) {
    case VAL_NULL:
        return null;

    case VAL_STRING:
        return readString();

    case VAL_INTEGER:
        return readInt();

    case VAL_MAP:
        return readHashMap(loader);

    ......
    }
}

這里對(duì)應(yīng)之前的writeValue(),先讀取之間寫(xiě)入的類(lèi)型常量值,如果是VAL_MAP,就調(diào)用readHashMap():

public final HashMap readHashMap(ClassLoader loader){
    int N = readInt();
    if (N < 0) {
        return null;
    }
    HashMap m = new HashMap(N);
    readMapInternal(m, N, loader);
    return m;
}

真相大白了,readHashMap()中直接new了一個(gè)HashMap,再依次讀取之前寫(xiě)入的K-V值,填充到HashMap中,所以B頁(yè)面拿到就是這個(gè)HashMap,而拿不到LinkedHashMap了。

一題多解

雖然不能直接傳LinkedHashMap,不過(guò)可以通過(guò)另一種方式來(lái)傳遞,那就是傳遞一個(gè)實(shí)現(xiàn)了Serializable接口的類(lèi)對(duì)象,將LinkedHashMap作為一個(gè)成員變量放入該對(duì)象中,再進(jìn)行傳遞。如:

public class MapWrapper implements Serializable {

  private HashMap mMap;

  public void setMap(HashMap map){
      mMap=map;
  }

  public HashMap getMap() {
      return mMap;
  }
}

那么為什么這樣傳遞就行了呢?其實(shí)也很簡(jiǎn)單,因?yàn)樵趙riteValue()時(shí),如果寫(xiě)入的是Serializable對(duì)象,那么就會(huì)調(diào)用writeSerializable():

public final void writeSerializable(Serializable s) {
    if (s == null) {
        writeString(null);
        return;
    }
    String name = s.getClass().getName();
    writeString(name);

    ByteArrayOutputStream baos = new ByteArrayOutputStream();
    try {
        ObjectOutputStream oos = new ObjectOutputStream(baos);
        oos.writeObject(s);
        oos.close();

        writeByteArray(baos.toByteArray());
    } catch (IOException ioe) {
        throw new RuntimeException("Parcelable encountered " +
            "IOException writing serializable object (name = " + name +
            ")", ioe);
    }
}

可見(jiàn)這里直接將這個(gè)對(duì)象給序列化成字節(jié)數(shù)組了,并不會(huì)因?yàn)槔锩姘粋€(gè)Map對(duì)象而再走入writeMap(),所以L(fǎng)inkedHashMap得以被保存了。

結(jié)論:

一句話(huà),遇到問(wèn)題就多看源碼!

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 227,572評(píng)論 6 531
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,071評(píng)論 3 414
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人,你說(shuō)我怎么就攤上這事。” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 175,409評(píng)論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀(guān)的道長(zhǎng)。 經(jīng)常有香客問(wèn)我,道長(zhǎng),這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 62,569評(píng)論 1 307
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 71,360評(píng)論 6 404
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 54,895評(píng)論 1 321
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 42,979評(píng)論 3 440
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 42,123評(píng)論 0 286
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,643評(píng)論 1 333
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 40,559評(píng)論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 42,742評(píng)論 1 369
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,250評(píng)論 5 356
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 43,981評(píng)論 3 346
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 34,363評(píng)論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 35,622評(píng)論 1 280
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 51,354評(píng)論 3 390
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 47,707評(píng)論 2 370

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

  • ¥開(kāi)啟¥ 【iAPP實(shí)現(xiàn)進(jìn)入界面執(zhí)行逐一顯】 〖2017-08-25 15:22:14〗 《//首先開(kāi)一個(gè)線(xiàn)程,因...
    小菜c閱讀 6,477評(píng)論 0 17
  • 從三月份找實(shí)習(xí)到現(xiàn)在,面了一些公司,掛了不少,但最終還是拿到小米、百度、阿里、京東、新浪、CVTE、樂(lè)視家的研發(fā)崗...
    時(shí)芥藍(lán)閱讀 42,314評(píng)論 11 349
  • 1. Java基礎(chǔ)部分 基礎(chǔ)部分的順序:基本語(yǔ)法,類(lèi)相關(guān)的語(yǔ)法,內(nèi)部類(lèi)的語(yǔ)法,繼承相關(guān)的語(yǔ)法,異常的語(yǔ)法,線(xiàn)程的語(yǔ)...
    子非魚(yú)_t_閱讀 31,707評(píng)論 18 399
  • Spring Cloud為開(kāi)發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見(jiàn)模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn),斷路器,智...
    卡卡羅2017閱讀 134,776評(píng)論 18 139
  • (四十二)商道之知損益知自省 道生一,一生二,二生三,三生萬(wàn)物。萬(wàn)物負(fù)陰而抱陽(yáng),沖氣以為和。人之所惡,唯孤寡不谷,...
    袖卷千重雪閱讀 414評(píng)論 0 1