放棄RxBus,擁抱RxJava(一):為什么避免使用EventBus/RxBus

EventBus和Otto在之前作為Android組件間通信工具,簡單方便十分受歡迎,但是也非常容易Abuse。大概有如下幾個缺點:

  • 由于是Event,在發布Event的時候就要做好準備可能并沒有人接受這個Event, Subscribe的時候也要做好準備可能永遠不會收到Event。Event無論順序還是時間上都某種程度上不太可控。如果你將數據寄托在Event上然后就直接在Android其他生命周期方法中直接使用這個數據或成員變量。那么很有可能你會得到NPE。
  • EventBus看似將你的程序解耦,但是又有些過了。我們常常使用EventBus傳數據,這已經是Dependency級別的數據而不是一個可以被解耦出來的模塊。這樣就造成了過多EventBus的代碼會造成代碼結構混亂,難以測試和追蹤,違背了解耦的初衷。這時如果有意或無意的造成了Nested Event。那情況會更糟。

由于EventBus的種種缺點,以及后面RxJava的出現。很多人都開始使用RxJava來取代EventBus。甚至Otto的官方介紹里都寫到:

Deprecated!

This project is deprecated in favor of RxJava and
RxAndroid. These projects permit the same event-driven
programming model as Otto, but they’re more capable and offer better control of threading.

If you’re looking for guidance on migrating from Otto to Rx, this post
is a good start.

鏈接是一個教你怎么使用RxJava來自己手動寫一個RxBus來代替EventBus的文章。雖然看起來是在用RxJava。但是實際上卻仍然在用EventBus。甚至這個封裝其實也并沒有GreenRobot或者Otto來的好。
我們看看Jake Wharton對RxBus的評價:


我想"RxBus"唯一的好處就是他是一個Rx的入門毒品。否則的話,你要么就不是在用Rx,要么你需要更加慣用的Rx資源 (渣翻譯見諒)

再來一個GitHub的:


subscribeActual部分我們先不考慮。然而Jake指出最好不要使用Relay來“重新發明”Event Bus.

這里看圖說話:
Jake Wharton在GOTO 2016 上的講座中提到,我們正常的Android編程是這樣的:



我們像一個中間人一樣。
而使用RxJava。 我們的結構,更像這樣



我們使用RxJava來直接把組件相連,對所接受到的數據作出反應,所謂的 "Reactive"。
而使用Eventbus? Jake 沒說, 我自己畫一個:

我們作為一個中間人,傳遞消息。EventBus作為另一個中間人。幫我們傳遞消息。(這也就是所謂的“看似解耦”)

再打個比方,雖然我們將EventBus翻譯成時間總線,但是其實總線就是Bus也就是公交車。而RxJava更像一個專車,Uber或者滴滴。他直接鏈接你的兩個或多個需要通信的類。傳輸數據,當然你可以做一個很大的專車,穿梭在所有類之間,也就是所謂的RxBus。所以在這里為什么放棄RxBus也就不言而喻了不是?

那么,問題來了?

怎樣才是正確(正常?)的RxJava使用方式?

其實Jake 也在GitHub的討論上給出了一個答案:


所以應該是,每當你想發布一個Event在EventBus時,直接暴露一個Observable出來。每當你想接受一個Event時,找到這個Observable并且Subscribe他。

這樣做的好處是什么?

  • 目標和地點都很明確。你的Subscriber明確的知道他Subscribe的是誰,而且明確的知道我需要作出什么反應。這也正是RxJava的核心“響應式編程”。
  • 由于使用了Observable,對于異常處理將會非常方便。而且還有功能強大全面的Operator來輔助你。
  • 雖然看起來耦合性有所增加。但是這是必要的,上面也說過,EventBus雖然在代碼上看似解耦。其實他們還是聯系在一起的。而我們這樣直接暴露Observable給需要的其他類,這完成了1 -> 1/N的鏈接,而不需要EventBus這個中間人來傳遞消息/事件,而且保證我們需要的事件一定會直接到達。

我們來舉個例子

上下兩個Fragment,上面的一個EditText,下面的一個TextView。上面的EditText變化的時候下面的TextView也跟著變化。

先把EditText的TextChangedListener封裝在Observable里:

        stringObservable = Observable.create(e -> editText.addTextChangedListener(new TextWatcher() {
            @Override
            public void beforeTextChanged(CharSequence s, int start, int count, int after) {

            }

            @Override
            public void onTextChanged(CharSequence s, int start, int before, int count) {
                e.onNext(s.toString());
            }

            @Override
            public void afterTextChanged(Editable s) {

            }
        }));

/**
***
*/
    //Expose Observable
    public Observable<String> getEditTextObservable() {
        return stringObservable;
    }```
不習慣自己封裝可以使用RxBinding :
    stringObservable = RxTextView.textChangeEvents(editText)
                                 .map(event -> event.text().toString());
再從我們的TextViewFragment中 取到這個封裝好的Observable:
@Override
public void onStart() {
    super.onStart();
    FragmentEditText fragment = (FragmentEditText) getFragmentManager().findFragmentByTag(FragmentEditText.TAG);
    if(fragment != null){
        fragment.getStringObservable().subscribe(s -> textView.setText(s));
    }
}
來看看效果:

![](http://upload-images.jianshu.io/upload_images/2417399-f008642da01d310f.gif?imageMogr2/auto-orient/strip)

當然,這里還有個問題
- 由于我們將editText封裝在Observable里,無論是create()方法還是使用RxBinding,都會持有這個View的強引用。造成內存泄漏。所以我們一定要在最后加入dispose()方法來釋放。所以我推薦使用RxBinding,他已經幫我們在dispose()方法里寫好了解除Listener的方法。
- 因為并沒有使用publish操作符,導致多個Subscriber的時候還是有些許問題??梢钥紤]直接加入.share().

具體我們如何講常用的數據/Callback封裝到Observable中。我會在接下來的文章中寫到。
>  [第二篇鏈接](http://www.lxweimin.com/p/d2df6bceeff9) 
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 230,527評論 6 544
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,687評論 3 429
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事?!?“怎么了?”我有些...
    開封第一講書人閱讀 178,640評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,957評論 1 318
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,682評論 6 413
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 56,011評論 1 329
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 44,009評論 3 449
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 43,183評論 0 290
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,714評論 1 336
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,435評論 3 359
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,665評論 1 374
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,148評論 5 365
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,838評論 3 350
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,251評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,588評論 1 295
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,379評論 3 400
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,627評論 2 380

推薦閱讀更多精彩內容

  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,764評論 25 708
  • 原文地址:http://gank.io/post/560e15be2dca930e00da1083 前言 我從去年...
    AFinalStone閱讀 2,221評論 5 23
  • 我從去年開始使用 RxJava ,到現在一年多了。今年加入了 Flipboard 后,看到 Flipboard 的...
    Jason_andy閱讀 5,561評論 7 62
  • 作者寄語 很久之前就想寫一個專題,專寫Android開發框架,專題的名字叫 XXX 從入門到放棄 ,沉淀了這么久,...
    戴定康閱讀 7,648評論 13 85
  • 我對你的喜歡, 如山澗里緩緩流淌地細流, 干凈,澄澈。 如八月天高高掛起的艷陽, 熱烈,奔放。 我對你的喜歡, 是...
    易個棟閱讀 662評論 0 2