Android:隨筆——RxJava的線程切換

轉(zhuǎn)載請(qǐng)標(biāo)明地址 QuincySx:[http://www.reibang.com/p/d9da64774f7b]


近期用到 RxJava ,線程切換的時(shí)候出了點(diǎn)小插曲仁讨,首先先上理論,在上實(shí)踐咐旧,不喜理論可跳過掂骏,此篇文章適合會(huì)使用 RxJava 的人群,如果還沒有接觸過可以自學(xué)過后再來讀這篇文章,這篇文章這幾個(gè)例子其實(shí)代碼都是基本都是一樣的眯娱,我也不知道這樣寫是不是更清晰

理論

總所周知 RxJava 在切換線程時(shí)用到了兩個(gè)方法 subscribeOn()observeOn() 下面來分別解釋一下這兩個(gè)方法

  • subscribeOn() : 影響的是最開始的被觀察者所在的線程礁苗。當(dāng)使用多個(gè) subscribeOn() 的時(shí)候,只有第一個(gè) subscribeOn() 起作用徙缴;
  • observeOn() : 影響的是跟在后面的操作(指定觀察者運(yùn)行的線程)试伙。所以如果想要多次改變線程,可以多次使用 observeOn于样;

我之前還看到有人說 subscribeOn() 必須在 observeOn() 的前面疏叨,不過經(jīng)過我測(cè)試他兩個(gè)的位置并沒有什么聯(lián)系,就如上面所說 第一次出現(xiàn) subscribeOn() 的地方是有效的穿剖,其他的無效


實(shí)踐

實(shí)踐一下蚤蔓,先來個(gè)基本的栗子


        Observable
                .create(new Observable.OnSubscribe<String>() {
                    @Override
                    public void call(Subscriber<? super String> subscriber) {
                        Log.e(TAG, "===create: " + Thread.currentThread().getName());
                        subscriber.onNext("1");
                    }
                })
                .map(new Func1<String, Integer>() {
                    @Override
                    public Integer call(String s) {
                        Log.e(TAG, "===String -> Integer: " + Thread.currentThread().getName());
                        return Integer.valueOf(s);
                    }
                })
                .flatMap(new Func1<Integer, Observable<String>>() {
                    @Override
                    public Observable<String> call(final Integer integer) {
                        Log.e(TAG, "===Integer->Observable: " + Thread.currentThread().getName());
                        return Observable.create(new Observable.OnSubscribe<String>() {
                            @Override
                            public void call(Subscriber<? super String> subscriber) {
                                Log.e(TAG, "===Observable<String> call: " + Thread.currentThread().getName());
                                for (int i = 0; i < integer; i++) {
                                    subscriber.onNext(i + "");
                                }
                                subscriber.onCompleted();
                            }
                        });
                    }
                })
                .map(new Func1<String, Long>() {
                    @Override
                    public Long call(String s) {
                        Log.e(TAG, "===String->Long: " + Thread.currentThread().getName());
                        return Long.parseLong(s);
                    }
                })
                .subscribeOn(Schedulers.io())
                .observeOn(AndroidSchedulers.mainThread())
                .subscribe(new Subscriber<Long>() {
                    @Override
                    public void onCompleted() {
                    }

                    @Override
                    public void onError(Throwable e) {
                    }

                    @Override
                    public void onNext(Long aLong) {
                        Log.e(TAG, "===onNext: " + Thread.currentThread().getName());
                    }
                });

這個(gè)例子呢就是簡(jiǎn)單的將 String 轉(zhuǎn)為 Integer 然后轉(zhuǎn)換發(fā)射源 發(fā)射 String 在然后 將 String 轉(zhuǎn)換為 Long 然后打印
這個(gè)例子呢沒有什么實(shí)際意義,只作為個(gè)示例
有沒有看到

.subscribeOn(Schedulers.io())
.observeOn(AndroidSchedulers.mainThread())

這兩句代碼 我切換了一下線程 糊余,接下來看一下我的運(yùn)行結(jié)果

===create: RxIoScheduler-2
===String -> Integer: RxIoScheduler-2
===Integer->Observable: RxIoScheduler-2
===Observable<String> call: RxIoScheduler-2
===String->Long: RxIoScheduler-2
===onNext: main

接下來解釋一下秀又,因?yàn)?code>subscribeOn(Schedulers.io())它指定了最開始的被觀察者所在的線程所以后面的操作都是根據(jù)最開始的被觀察者制定的線程運(yùn)行的,又因?yàn)?.observeOn(AndroidSchedulers.mainThread()) 它指定了都面的操作符使用主線程運(yùn)行贬芥。


接下來再寫一個(gè)多 subscribeOn() observeOn()的情況

      Observable
                .create(new Observable.OnSubscribe<String>() {
                    @Override
                    public void call(Subscriber<? super String> subscriber) {
                        Log.e(TAG, "===create: " + Thread.currentThread().getName());
                        subscriber.onNext("1");
                    }
                })
                .subscribeOn(AndroidSchedulers.mainThread())
                .observeOn(Schedulers.io())
                .map(new Func1<String, Integer>() {
                    @Override
                    public Integer call(String s) {
                        Log.e(TAG, "===String -> Integer: " + Thread.currentThread().getName());
                        return Integer.valueOf(s);
                    }
                })
                .observeOn(AndroidSchedulers.mainThread())
                .flatMap(new Func1<Integer, Observable<String>>() {
                    @Override
                    public Observable<String> call(final Integer integer) {
                        Log.e(TAG, "===Integer->Observable: " + Thread.currentThread().getName());
                        return Observable.create(new Observable.OnSubscribe<String>() {
                            @Override
                            public void call(Subscriber<? super String> subscriber) {
                                Log.e(TAG, "===Observable<String> call: " + Thread.currentThread().getName());
                                for (int i = 0; i < integer; i++) {
                                    subscriber.onNext(i + "");
                                }
                                subscriber.onCompleted();
                            }
                        });
                    }
                })
                .observeOn(Schedulers.io())
                .map(new Func1<String, Long>() {
                    @Override
                    public Long call(String s) {
                        Log.e(TAG, "===String->Long: " + Thread.currentThread().getName());
                        return Long.parseLong(s);
                    }
                })
                .subscribeOn(Schedulers.io())
                .observeOn(AndroidSchedulers.mainThread())
                .subscribe(new Subscriber<Long>() {
                    @Override
                    public void onCompleted() {
                    }

                    @Override
                    public void onError(Throwable e) {
                    }

                    @Override
                    public void onNext(Long aLong) {
                        Log.e(TAG, "===onNext: " + Thread.currentThread().getName());
                    }
                });

運(yùn)行結(jié)果

===create: main
===String -> Integer: RxIoScheduler-4
===Integer->Observable: main
===Observable<String> call: main
===String->Long: RxIoScheduler-3
===onNext: main

下面應(yīng)該不用我解釋就能知道結(jié)果為什么是這樣的
簡(jiǎn)單解釋一下 因?yàn)橛袃蓚€(gè)subscribeOn()所以取第一個(gè)吐辙,所以最開始的被觀察者所在的線程為主線程,接著使用observeOn()使用制定后面的操作符為 io 線程蘸劈,接著observeOn()又指定后面的操作符為主線程...以此類推不再贅述


接下來到了我要說的重點(diǎn)了也是我受到迷惑的地方昏苏,我就簡(jiǎn)單的寫一段栗子來重現(xiàn)一下,直接上代碼

      Observable
                .create(new Observable.OnSubscribe<String>() {
                    @Override
                    public void call(Subscriber<? super String> subscriber) {
                        Log.e(TAG, "===create: " + Thread.currentThread().getName());
                        subscriber.onNext("1");
                    }
                })
                .map(new Func1<String, Integer>() {
                    @Override
                    public Integer call(String s) {
                        Log.e(TAG, "===String -> Integer: " + Thread.currentThread().getName());
                        return Integer.valueOf(s);
                    }
                })
                .flatMap(new Func1<Integer, Observable<String>>() {
                    @Override
                    public Observable<String> call(final Integer integer) {
                        Log.e(TAG, "===Integer->Observable: " + Thread.currentThread().getName());
                        return forEach(integer);
                    }
                })
                .map(new Func1<String, Long>() {
                    @Override
                    public Long call(String s) {
                        Log.e(TAG, "===String->Long: " + Thread.currentThread().getName());
                        return Long.parseLong(s);
                    }
                })
                .subscribeOn(Schedulers.io())
                .observeOn(AndroidSchedulers.mainThread())
                .subscribe(new Subscriber<Long>() {
                    @Override
                    public void onCompleted() {
                    }

                    @Override
                    public void onError(Throwable e) {
                    }

                    @Override
                    public void onNext(Long aLong) {
                        Log.e(TAG, "===onNext: " + Thread.currentThread().getName());
                    }
                });

可以看到 flatMap 那個(gè)操作符哪里使用了一個(gè)方法 forEach(int) 代碼如下

 public Observable<String> forEach(final int integer) {
        return Observable
                .create(new Observable.OnSubscribe<String>() {
                    @Override
                    public void call(Subscriber<? super String> subscriber) {
                        Log.e(TAG, "===Observable<String> call: " + Thread.currentThread().getName());
                        for (int i = 0; i < integer; i++) {
                            subscriber.onNext(i + "");
                        }
                        subscriber.onCompleted();
                    }
                })
                .subscribeOn(Schedulers.io())
                .observeOn(AndroidSchedulers.mainThread());
    }

再貼一下結(jié)果

===create: RxIoScheduler-2
===String -> Integer: RxIoScheduler-2
===Integer->Observable: RxIoScheduler-2
===Observable<String> call: RxIoScheduler-3
===String->Long: main
===onNext: main

解釋一下結(jié)果:我想大家肯定發(fā)現(xiàn)了端倪,為什么 String -> Long 這個(gè)步驟為什么會(huì)在主線程里運(yùn)行呢捷雕,原因呢很簡(jiǎn)單 flatMap 變換被監(jiān)聽者椒丧,這個(gè)被監(jiān)聽者使用observeOn切換了后邊操作符的線程,影響到了 flatMap 后面的 map 操作符所以導(dǎo)致了如此結(jié)果

緊接著我們?cè)诳匆粋€(gè)例子

Observable
                .create(new Observable.OnSubscribe<String>() {
                    @Override
                    public void call(Subscriber<? super String> subscriber) {
                        Log.e(TAG, "===create: " + Thread.currentThread().getName());
                        subscriber.onNext("1");
                    }
                })
                .map(new Func1<String, Integer>() {
                    @Override
                    public Integer call(String s) {
                        Log.e(TAG, "===String -> Integer: " + Thread.currentThread().getName());
                        return Integer.valueOf(s);
                    }
                })
                .flatMap(new Func1<Integer, Observable<String>>() {
                    @Override
                    public Observable<String> call(final Integer integer) {
                        Log.e(TAG, "===Integer->Observable: " + Thread.currentThread().getName());
                        return Observable.create(new Observable.OnSubscribe<String>() {
                            @Override
                            public void call(final Subscriber<? super String> subscriber) {
                                forEach(integer, new CallBack() {
                                    @Override
                                    public void call(String s) {
                                        Log.e(TAG, "===Subscriber: " + Thread.currentThread().getName());
                                        subscriber.onNext(s);
                                    }
                                });
                            }
                        });
                    }
                })
                .map(new Func1<String, Long>() {
                    @Override
                    public Long call(String s) {
                        Log.e(TAG, "===String->Long: " + Thread.currentThread().getName());
                        return Long.parseLong(s);
                    }
                })
                .subscribeOn(Schedulers.io())
                .observeOn(AndroidSchedulers.mainThread())
                .subscribe(new Subscriber<Long>() {
                    @Override
                    public void onCompleted() {
                    }

                    @Override
                    public void onError(Throwable e) {
                    }

                    @Override
                    public void onNext(Long aLong) {
                        Log.e(TAG, "===onNext: " + Thread.currentThread().getName());
                    }
                });
public void forEach(final int integer, final CallBack back) {
        Observable
                .create(new Observable.OnSubscribe<String>() {
                    @Override
                    public void call(Subscriber<? super String> subscriber) {
                        Log.e(TAG, "===Observable<String> call: " + Thread.currentThread()
                                .getName());
                        for (int i = 0; i < integer; i++) {
                            subscriber.onNext(i + "");
                        }
                        subscriber.onCompleted();
                    }
                })
                .subscribeOn(Schedulers.io())
                .observeOn(AndroidSchedulers.mainThread())
                .subscribe(new Action1<String>() {
                    @Override
                    public void call(String s) {
                        back.call(s);
                    }
                });
    }

    interface CallBack {
        void call(String s);
    }

結(jié)果

===create: RxIoScheduler-4
===String -> Integer: RxIoScheduler-4
===Integer->Observable: RxIoScheduler-4
===Observable<String> call: RxIoScheduler-5
===Subscriber: main
===String->Long: main
===onNext: main

大家發(fā)現(xiàn)的現(xiàn)在還是 和上一個(gè)例子相仿救巷,主要的改變還是在 flatMap() 這個(gè)方法里壶熏,現(xiàn)在我的做法是創(chuàng)建一個(gè)被監(jiān)聽者,然后里面是調(diào)用forEach()等待接口返回值浦译,再往下發(fā)射數(shù)據(jù)棒假。
注意以下幾點(diǎn):

  1. forEach()方法里面切換了線程,這個(gè)回調(diào)接口的線程為 主線程
  2. 我們?cè)?code>flatMap()這里新建了一個(gè)發(fā)射源(被監(jiān)聽者)精盅、在這里我們并沒有指定它的線程帽哑,所以flatMap()是什么線程,這個(gè)被監(jiān)聽者就是什么線程

看清以上兩點(diǎn)叹俏,我們分析一下妻枕,flatMap()以上的結(jié)果大家肯定都能想出來,我們就直接從flatMap()開始分析了粘驰,都知道了回調(diào)的接口里面的線程為主線程屡谐,那么它作為被觀察者它下面的操作符按照 RxJava 線程切換的基本原理來說,肯定也是主線程蝌数。

上個(gè)圖理解一下

所以大家如果像我這樣使用 flatMap() 的時(shí)候一定注意下面操作符的線程

總結(jié)

今天去調(diào)一個(gè) Js 的方法愕掏,然后有一個(gè)接口回調(diào),無意中看見一個(gè) 名字奇怪的 線程名稱顶伞,一想 RxJava 的 io 線程名 應(yīng)該是 RxIoScheduler 開頭的啊饵撑,突然有點(diǎn)蒙,感覺自己一點(diǎn)都不了解 RxJava 的線程 切換了唆貌,結(jié)果分析一波 也是萬變不離其宗滑潘,還是挺有意思的一個(gè)小插曲

歡迎各位前來拍磚

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市挠锥,隨后出現(xiàn)的幾起案子众羡,更是在濱河造成了極大的恐慌,老刑警劉巖蓖租,帶你破解...
    沈念sama閱讀 217,542評(píng)論 6 504
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件粱侣,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡蓖宦,警方通過查閱死者的電腦和手機(jī)齐婴,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,822評(píng)論 3 394
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來稠茂,“玉大人柠偶,你說我怎么就攤上這事情妖。” “怎么了诱担?”我有些...
    開封第一講書人閱讀 163,912評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵毡证,是天一觀的道長。 經(jīng)常有香客問我蔫仙,道長料睛,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,449評(píng)論 1 293
  • 正文 為了忘掉前任摇邦,我火速辦了婚禮恤煞,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘施籍。我一直安慰自己居扒,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,500評(píng)論 6 392
  • 文/花漫 我一把揭開白布丑慎。 她就那樣靜靜地躺著喜喂,像睡著了一般。 火紅的嫁衣襯著肌膚如雪竿裂。 梳的紋絲不亂的頭發(fā)上夜惭,一...
    開封第一講書人閱讀 51,370評(píng)論 1 302
  • 那天,我揣著相機(jī)與錄音铛绰,去河邊找鬼。 笑死产喉,一個(gè)胖子當(dāng)著我的面吹牛捂掰,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播曾沈,決...
    沈念sama閱讀 40,193評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼这嚣,長吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了塞俱?” 一聲冷哼從身側(cè)響起姐帚,我...
    開封第一講書人閱讀 39,074評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎障涯,沒想到半個(gè)月后罐旗,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,505評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡唯蝶,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,722評(píng)論 3 335
  • 正文 我和宋清朗相戀三年九秀,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片粘我。...
    茶點(diǎn)故事閱讀 39,841評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡鼓蜒,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情都弹,我是刑警寧澤娇豫,帶...
    沈念sama閱讀 35,569評(píng)論 5 345
  • 正文 年R本政府宣布,位于F島的核電站畅厢,受9級(jí)特大地震影響冯痢,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜或详,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,168評(píng)論 3 328
  • 文/蒙蒙 一系羞、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧霸琴,春花似錦椒振、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,783評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至选调,卻和暖如春夹供,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背仁堪。 一陣腳步聲響...
    開封第一講書人閱讀 32,918評(píng)論 1 269
  • 我被黑心中介騙來泰國打工哮洽, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人弦聂。 一個(gè)月前我還...
    沈念sama閱讀 47,962評(píng)論 2 370
  • 正文 我出身青樓鸟辅,卻偏偏與公主長得像,于是被迫代替她去往敵國和親莺葫。 傳聞我的和親對(duì)象是個(gè)殘疾皇子匪凉,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,781評(píng)論 2 354

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