国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 學院 > 開發設計 > 正文

Intent傳遞數據底層分析

2019-11-08 00:19:16
字體:
來源:轉載
供稿:網友

我們知道在Activity切換時,如果需要向下一個ActivityB傳遞數據,可以借助Intent對象的putExtra方法。 但是不知各位有沒有想過這樣一個問題:ActivityB中獲取到的對象跟上一個Activity中的那個對象有什么關系? 換句話說就是,我在ActivityB中通過Intent獲取的對象跟ActivityA中的那個對象,有沒有可能是同一個對象? 按照常理來說,博主提出一個設想后續的就是證明過程了,但是我要遺憾的告訴你,這里并非是同一個對象。(PS:廢話,如果是同一個對象,那還有EventBus這些東西什么事兒 T_T) 那么問題又來了,這兩個Activity都在同一個進程里面,甚至都在同一個線程里面,數據本來就是可以共享的,為什么從一個Activity傳到另一個Activity之后,就不是一個對象了呢?它從什么時候變成另外的對象的呢? 不著急,且聽我慢慢道來。

Intent是什么東西?

public class Intent implements Parcelable, Cloneable

上面是Intent類的完整聲明,可以知道它實現了Parcelable接口。Parcelable接口是什么呢?這東西是Android上專門用來對數據進行序列化的,并且在跨進程通訊時Parceable對象是可以直接傳輸的。 接下來我們來看看將數據放入Intent的時,做了哪些處理。 以String為例,先看putExtra方法的代碼

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

很簡單,就是將數據放入mExtras這個Bundle對象中,順便說一句Bundle類也實現了Parcelable接口。繼續往下跟代碼

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

內部就是將數據放入一個Map中保存。到這里數據放入Intent的過程就完成了,實際上就是Intent中有一個Bundle對象,而這個Bundle對象中又有一個Map,然后數據就保存在這里。至于那個unparcel()方法與我們的分析過程無關,有興趣的讀者可以去研究一下。 然后,我們再看取數據的過程。 繼續以String作為例子,看Intent中的代碼

public String getStringExtra(String name) { return mExtras == null ? null : mExtras.getString(name); }

mExtras應該很熟悉了,這是個Bundle對象,剛剛保存數據的時候就是把數據保存在它里面的。再看它的getString方法

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

就是直接從Map里面拿出我們之前保存的String,try語句只是在驗證取出的數據是否為String類型。 那照這么分析的話,兩個Activity中的對象應該就是同一個對象才對啊!!為什么又說不是同一個對象呢?

為什么不是同一個對象?

如果你在putExtra之后,馬上又getExtra出來,那么你取出來的對象肯定是同一個對象,這個沒錯! 但是這里我們要注意兩點: 1.Intent中允許保存的數據類型是有限制的,準確的說是Bundle的限制,因為實質上數據是保存在Bundle中。如果我們要保存自己定義的對象,那么我們的對象必須實現了Parcelable接口或者Serializable接口。 2.我們使用Intent的方式,基本都是在一個Activity中存入,然后從另一個Activity中取出。 那么問題很明顯就出在Activity的啟動過程了。詳細的啟動過程大家可以參考老羅的文章Activity啟動過程。 這里大概說一下,首先我們的app運行在app自己的進程appPRocess中,然后系統在啟動的時候會啟動一個系統進程systemProcess。而在Activity啟動時,需要向一個叫做ActivityManagerService的系統服務去注冊,這樣我們的Activity才能有生命周期的回調。這個ActivityManagerService服務就運行在systemProcess中。注冊完之后,再回到appProcess中,完成新Activity的啟動。在這個注冊過程中,我們的intent是全程參與的。 說到這里就明白了,當我們調用startActivity(intent)啟動另外的Activity的時候,我們的intent已經完成了兩次跨進程通信,而它里面的對象已經經歷了兩輪序列化和反序列化,肯定不可能是同一個對象了。

這里順便說一個問題:為什么Serializable也可以跨進程傳輸? 熟悉AIDL的同學都很清楚,AIDL跨進程通信支持的數據類型是:

java 的原生類型,如int,boolean,long,float…String 和CharSequenceList 和 Map ,List和Map 對象的元素必須是AIDL支持的數據類型AIDL 自動生成的接口 需要導入(import)實現android.os.Parcelable 接口的類. 需要導入(import)。

這里并不包括Serializable類型。 于是去看了源碼,發現是Parcel自己對Serializable類型的對象做了兼容,可以直接寫入其中。

頂0踩
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 苏尼特左旗| 荃湾区| 深泽县| 灵璧县| 桃源县| 嘉峪关市| 定南县| 梅州市| 永州市| 娱乐| 宁陵县| 洪雅县| 梁河县| 赤城县| 宁强县| 砀山县| 北海市| 手机| 永顺县| 徐汇区| 塘沽区| 沾益县| 安福县| 秦皇岛市| 西林县| 馆陶县| 公主岭市| 宽甸| 建德市| 通道| 客服| 昌吉市| 武清区| 南涧| 浦县| 玉田县| 高唐县| 健康| 介休市| 东山县| 通渭县|