EventBus 3.0 源碼分析

功能

EventBus 是一個 Android 事件發(fā)布/訂閱框架居暖,通過解耦發(fā)布者和訂閱者簡化 Android 事件傳遞契邀。事件傳遞既可用于 Android 四大組件間通訊奈辰,也可以用戶異步線程和主線程間通訊等等忙上。傳統(tǒng)的事件傳遞方式包括:Handler藕漱、BroadCastReceiver、Interface 回調(diào)臀稚,相比之下 EventBus 的優(yōu)點(diǎn)是代碼簡潔市袖,使用簡單,并將事件發(fā)布和訂閱充分解耦烁涌。

使用

01.png

這是GitHub,EventBus的使用介紹酒觅,用起來還是很簡單的撮执。

類圖

02.png

從上面的類圖可以看出,都是圍繞類EventBus 通過組合來建立關(guān)聯(lián)舷丹。
可以看到優(yōu)秀的框架設(shè)計(jì)抒钱,多用組合,少用繼承颜凯。

源碼分析

EventBus.getDefault()谋币;

獲取Event Bus的實(shí)例:

03.png

使用單例模式來創(chuàng)建出Event Bus對象,采用了雙重驗(yàn)證 以保證線程安全症概。蕾额,看看Event Bus的構(gòu)造函數(shù)。

04.png

在構(gòu)造函數(shù)里進(jìn)行了一系列的初始化工作彼城,主要通過類EventBusBuilder 使用Build的模式來初始化EventBus的一些配置诅蝶。

還有幾個重要的數(shù)據(jù)結(jié)構(gòu)集合的初始化:

subscriptionsByEventType

訂閱該事件的所有訂閱者 退个。是一個以 key:訂閱的事件 value:訂閱這個事件的所有訂閱者。因此當(dāng)post的時候會遍歷這個Map相應(yīng)對應(yīng)事件的List调炬。

typesBySubscriber

訂閱者中的所有訂閱事件 语盈。是一個以key:訂閱者 value:訂閱者中的所有訂閱事件。用于后續(xù)的取消訂閱缰泡。

三個Poster類

05.png

Poster類是用來處理sticky事件刀荒。

register(this)

06.png

委托SubscriberMethodFinder類,查找出訂閱者中的所有訂閱事件方法棘钞。進(jìn)入findSubscriberMethods方法看看:

07.png

該方法的處理流程大概是這樣的:

09.png

這里我們只分析 通過findUsingReflection 方法缠借,通過反射獲取訂閱者中的所有訂閱事件方法。進(jìn)去看看這個方法:

11.png

查找到的訂閱事件會在臨時的類FindState中做校驗(yàn)和保存武翎,為什么會提到這個臨時類呢烈炭?這里有一個很好的設(shè)計(jì),使用了“對象池”的概念來創(chuàng)建FindState對象宝恶,和線程池同一個概念符隙, 這樣可以使對象FindState復(fù)用,防止創(chuàng)建過多的對象垫毙,增加內(nèi)存開銷霹疫。

去看看如何復(fù)用FindState對象的,進(jìn)入上圖prepareFindState方法:

12.png

在112中 可以看到 會從對象池FIND_STATE_POOL取出FindState 而不是每次都創(chuàng)建對象综芥。

回到 findUsingReflection方法當(dāng)查找完所有的訂閱事件方法丽蝎,調(diào)用getMethodsAndRelease對FindState進(jìn)行回收復(fù)用:

13.png

還是回到 findUsingReflection 方法,調(diào)用了findUsingReflectionInSingleClass 方法來進(jìn)行查找訂閱事件方法膀藐,進(jìn)去看看是怎么查找的:

14.png
  1. 在154行 通過反射 獲取訂閱者(this)的所有方法屠阻。
  2. 在160 行遍歷這些方法。
  3. 在164行 判斷是否帶@Subscribe 注解的方法 是否只有一個參數(shù)额各,如果否 就會拋出異常
  4. 在165行判斷方法是否帶@Subscribe 注解 国觉,如果是,通過FindState對象進(jìn)行校驗(yàn)和保存虾啦。

其實(shí)很簡單 就是通過反射來查找到訂閱者的所有方法麻诀,查找出帶@Subscribe注解的方法保存起來。

回到register方法傲醉,查找完所有的訂閱者中的所有訂閱事件方法蝇闭,之后遍歷列表,進(jìn)入subscribe方法:

16.png
  1. 在146行 獲取訂閱方法的訂閱類型(就是帶@Subscribe 注解的方法的第一個參數(shù))硬毕。
  2. 封裝Subscription對象呻引,保存到subscriptionsByEventType數(shù)據(jù)集合中。
  3. 在159行吐咳,根據(jù)優(yōu)先級 把封裝Subscription對象苞七,保存到typesBySubscriber數(shù)據(jù)集合中藐守。
  4. 在174行 如果是sticky事件,立即post sticky事件到當(dāng)前訂閱者蹂风。

整個register的流程大概就是這樣:

17.png

post 事件

最后通過post方法 來發(fā)送事件卢厂,進(jìn)入post方法

18.png

首先從currentPostingThreadState 獲取 PostingThreadState 對象

currentPostingThreadState 是個ThreadLocal對象

19.png

ThreadLocal使得各線程能夠保持各自獨(dú)立的一個對象。
把要post的事件加入PostingThreadState 的eventQueue隊(duì)列中惠啄,循環(huán)取出事件慎恒。
進(jìn)入postSingleEvent方法

20.png
  1. 在365行,判斷是否觸發(fā)訂閱該事件的父類撵渡,接口的方法融柬。
  2. 調(diào)用postSingleEventForEventType 方法分發(fā)事件。

進(jìn)入postSingleEventForEventType方法

21.png
  1. 在389行趋距,從subscriptionsByEventType數(shù)據(jù)集合中取出該訂閱事件的所有訂閱者粒氧。
  2. 在392行,分發(fā)所有的訂閱者节腐,通過調(diào)用postToSubscription方法外盯。

進(jìn)入postToSubscription方法

22.png

根據(jù)不同的threadMode 在不同的線程中處理,最終都會調(diào)用invokeSubscriber方法翼雀,把事件invoke到訂閱者的方法中饱苟。

23.png

ThreadMode 共有四類:

  1. PostThread:默認(rèn)的 ThreadMode,表示在執(zhí)行 Post 操作的線程直接調(diào)用訂閱者的事件響應(yīng)方法狼渊,不論該線程是否為主線程(UI 線程)箱熬。當(dāng)該線程為主線程時,響應(yīng)方法中不能有耗時操作狈邑,否則有卡主線程的風(fēng)險(xiǎn)城须。適用場景:對于是否在主線程執(zhí)行無要求,但若 Post 線程為主線程米苹,不能耗時的操作酿傍;
  2. MainThread:在主線程中執(zhí)行響應(yīng)方法。如果發(fā)布線程就是主線程驱入,則直接調(diào)用訂閱者的事件響應(yīng)方法,否則通過主線程的 Handler 發(fā)送消息在主線程中處理——調(diào)用訂閱者的事件響應(yīng)函數(shù)氯析。顯然亏较,MainThread
    類的方法也不能有耗時操作,以避免卡主線程掩缓。適用場景:必須在主線程執(zhí)行的操作雪情;
  3. BackgroundThread:在后臺線程中執(zhí)行響應(yīng)方法。如果發(fā)布線程不是主線程你辣,則直接調(diào)用訂閱者的事件響應(yīng)函數(shù)巡通,否則啟動唯一的后臺線程去處理尘执。由于后臺線程是唯一的,當(dāng)事件超過一個的時候宴凉,它們會被放在隊(duì)列中依次執(zhí)行誊锭,因此該類響應(yīng)方法雖然沒有PostThread類和MainThread類方法對性能敏感,但最好不要有重度耗時的操作或太頻繁的輕度耗時操作弥锄,以造成其他操作等待丧靡。適用場景:操作輕微耗時且不會過于頻繁,即一般的耗時操作都可以放在這里籽暇;
  4. Async:不論發(fā)布線程是否為主線程温治,都使用一個空閑線程來處理。和BackgroundThread不同的是戒悠,Async類的所有線程是相互獨(dú)立的熬荆,因此不會出現(xiàn)卡線程的問題。適用場景:長耗時操作绸狐,例如網(wǎng)絡(luò)訪問卤恳。

引用出自

unregister

25.png

從 typesBySubscriber 數(shù)據(jù)集合中取出該訂閱者的所有訂閱方法,remove所有訂閱者的訂閱方法六孵,防止內(nèi)存泄漏纬黎。

整個post流程:

24.png

最后

EventBus 屬于一個比較容易理解的開源庫,項(xiàng)目整體的框架設(shè)計(jì)采用了觀察者模式劫窒,但不論從使用方式和實(shí)現(xiàn)方式上都是非常值得我們學(xué)習(xí)的開源項(xiàng)目本今。比如,對象池的設(shè)計(jì)主巍,三大Poster類的設(shè)計(jì)冠息,多用組合,少用繼承孕索,緩存的設(shè)計(jì)等都值得我們借鑒和使用逛艰。

END。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末搞旭,一起剝皮案震驚了整個濱河市散怖,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌肄渗,老刑警劉巖镇眷,帶你破解...
    沈念sama閱讀 216,591評論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異翎嫡,居然都是意外死亡欠动,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,448評論 3 392
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來具伍,“玉大人翅雏,你說我怎么就攤上這事∪搜浚” “怎么了望几?”我有些...
    開封第一講書人閱讀 162,823評論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長啼肩。 經(jīng)常有香客問我橄妆,道長,這世上最難降的妖魔是什么祈坠? 我笑而不...
    開封第一講書人閱讀 58,204評論 1 292
  • 正文 為了忘掉前任害碾,我火速辦了婚禮,結(jié)果婚禮上赦拘,老公的妹妹穿的比我還像新娘慌随。我一直安慰自己,他們只是感情好躺同,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,228評論 6 388
  • 文/花漫 我一把揭開白布阁猜。 她就那樣靜靜地躺著,像睡著了一般蹋艺。 火紅的嫁衣襯著肌膚如雪剃袍。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,190評論 1 299
  • 那天捎谨,我揣著相機(jī)與錄音民效,去河邊找鬼。 笑死涛救,一個胖子當(dāng)著我的面吹牛畏邢,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播检吆,決...
    沈念sama閱讀 40,078評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼舒萎,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了蹭沛?” 一聲冷哼從身側(cè)響起臂寝,我...
    開封第一講書人閱讀 38,923評論 0 274
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎摊灭,沒想到半個月后咆贬,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,334評論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡斟或,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,550評論 2 333
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了集嵌。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片萝挤。...
    茶點(diǎn)故事閱讀 39,727評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡御毅,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出怜珍,到底是詐尸還是另有隱情端蛆,我是刑警寧澤,帶...
    沈念sama閱讀 35,428評論 5 343
  • 正文 年R本政府宣布酥泛,位于F島的核電站今豆,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏柔袁。R本人自食惡果不足惜呆躲,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,022評論 3 326
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望捶索。 院中可真熱鬧插掂,春花似錦、人聲如沸腥例。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,672評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽燎竖。三九已至璃弄,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間构回,已是汗流浹背夏块。 一陣腳步聲響...
    開封第一講書人閱讀 32,826評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留捐凭,地道東北人拨扶。 一個月前我還...
    沈念sama閱讀 47,734評論 2 368
  • 正文 我出身青樓,卻偏偏與公主長得像茁肠,于是被迫代替她去往敵國和親患民。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,619評論 2 354

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

  • 我每周會寫一篇源代碼分析的文章,以后也可能會有其他主題.如果你喜歡我寫的文章的話,歡迎關(guān)注我的新浪微博@達(dá)達(dá)達(dá)達(dá)s...
    SkyKai閱讀 24,927評論 23 184
  • 對于Android開發(fā)老司機(jī)來說肯定不會陌生垦梆,它是一個基于觀察者模式的事件發(fā)布/訂閱框架匹颤,開發(fā)者可以通過極少的代碼...
    飛揚(yáng)小米閱讀 1,475評論 0 50
  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,077評論 25 707
  • 我覺得自己真的活的有點(diǎn)累了印蓖,有點(diǎn)不知道該怎么繼續(xù)去努力。 從小我都是一個乖到不行的小姑娘京腥。 小學(xué)的時候赦肃,大家都一樣...
    yangming14閱讀 277評論 0 0
  • 做了一些404頁面練習(xí),根據(jù)網(wǎng)上一些資料以及結(jié)合自己的理解,來總結(jié)一下關(guān)于404有頁面的一些東西他宛。 404頁面是什...
    青木睡不著閱讀 521評論 0 1