RxJava操作符系列一

detail.jpg

前言

第一次接觸學習RxJava應該是一兩個月前的事情了,但其中也是斷斷續續,最近又再次去學習RxJava,和當初剛接觸RxJava完全不是同樣的心情,輕松了很多,也感受到了RxJava的魅力,真是不由衷感嘆太牛了。目前關于RxJava的文章也很多,個人推薦兩篇扔物線的給 Android 開發者的 RxJava 詳解和大頭鬼Bruce的譯文 深入淺出RxJava系列。那么這篇文章通過代碼介紹RxJava中的操作符,以及操作符的使用。當然操作符較多,準備分幾篇文章介紹。如果你想提前學習其他操作符可以去GitHub,歡迎star項目。


RxJava操作符源碼傳送門


基礎介紹

為了力求有沒有RxJava基礎都能看懂此文,簡單介紹一下RxJava以及一些名詞。在RxJava開源的Github上是這樣解釋的a library for composing asynchronous and event-based programs using observable sequences for the Java VM。無論多么復雜的邏輯,都可以保持整潔的代碼格式。

在RxJava中最重要的就是Observable(被觀察者),subscribe(訂閱),Observer(觀察者)或者Subscriber(訂閱者),Observable也就是數據(事件)源,Subscriber負責接收以及處理數據(事件)。當然要想實現兩者通信,需要有一種機制那就是訂閱。Observer 通過 subscribe() 方法實現訂閱關系,從而 Observable 可以在需要的時候發出事件來通知 Observer。

例如張三(觀察者)想看某款新聞軟件的科技信息(被觀察者),由于科技信息是每天推送或者不定時推送,如果張三一直盯著手機屏幕看并且刷新消息是不是又新的信息,顯然不現實。這時候就可以通過張三 subscribe(訂閱)科技信息,而實現當有新的科技信息時自動給張三推送消息,在這期間,張三并不需要一直盯著屏幕刷新聞。在我們平時的認知中實現訂閱應該是張三.subscribe(科技新聞),不過在RxJava代碼中實現訂閱應該寫成科技新聞.subscribe(張三)。

在RxJava中,有三個事件回調方法,分別是onNext(),OnError(),onCompleted(),onNext()是最終輸出及處理數據的回調,在發射數據過程中出現錯誤異常會回調OnError()方法,當不會再有新的 onNext() 發出時,需要觸發 onCompleted() 方法作為標志。,OnError()和onCompleted()是互斥的。下面舉一個最簡單的例子

        Observable observable2 = Observable.just("也許當初忙著微笑和哭泣", "忙著追逐天空中的流星", "人理所當然的忘記", "是誰風里雨里一直默默守護在原地");
        Subscriber subscriber = new Subscriber<String>() {
            @Override
            public void onCompleted() {
                Log.e(TAG, "onCompleted: " )
            }

            @Override
            public void onError(Throwable e) {
                Log.e(TAG, "onError: ")
            }

            @Override
            public void onNext(String s) {
               Log.e(TAG, "onNext: "+s )
            }
        };
        observable.subscribe(subscriber);

運行后打印信息為

onNext: 也許當初忙著微笑和哭泣
onNext: 忙著追逐天空中的流星
onNext: 人理所當然的忘記
onNext: 是誰風里雨里一直默默守護在原地
onCompleted: 

Create

我們可以使用該操作符從零開始創建一個Observable,給這個操作符傳遞一個接受觀察者作為參數的函數,并調用觀察者的onNext,onError和onCompleted方法。如下

        //被觀察者
        Observable<String> observable = Observable.create(new Observable.OnSubscribe<String>() {
            @Override
            public void call(Subscriber<? super String> subscriber) {
                //可以多次調用subscriber.onNext("大家好")發射數據
                subscriber.onNext("大家好");
                subscriber.onNext("我開始學習RxJava");
                subscriber.onCompleted();

            }
        });

發送數據需要在毀掉方法call中調用subscriber的onNext(),onNext(T)發送的參數需要和Observable.OnSubscribe<T>()中參數相同,在上面我們傳入的是String類型。創建后Observale后,我們需要創建Subscriber(觀察者)去處理observable發送的數據。如下

        Subscriber<String> subscriber = new Subscriber<String>() {
            @Override
            public void onCompleted() {
                Log.e(TAG, "onCompleted");
            }
            @Override
            public void onError(Throwable e) {
                Log.e(TAG, e.getMessage());
            }
            @Override
            public void onNext(String s) {
                Log.e(TAG, "onNext:"+s);
            }
        };

數據成功發送后,會回調Subscriber的onNext()的方法,其中的參數就是接收到的數據。當onNext()接收數據完畢后會執行onCompleted(),如果中途有環節出現錯誤異常,會執行onError()。現在觀察者和被觀察者都創建完畢了,他們執行還需要一個前提就是訂閱,如果不訂閱,observable并不會發射數據,subscribe也不會接收數據,訂閱代碼如下

observable.subscribe(subscriber);

執行后輸出信息

onNext:大家好
onNext:我開始學習RxJava
onCompleted

from

該操作符是將其它種類的對象和數據類型轉換為Observable,如果當你發射的的數據是同一種類型,而不是混合使用Observables和其它類型的數據,會非常方便。如下創建Observable

        Integer[] integers = {1,2, 3, 4};
        Observable<Integer> observable=Observable.from(integers);
         Subscriber<String> subscriber = new Subscriber<Integer>() {
            @Override
            public void onCompleted() {
                Log.e(TAG, "onCompleted");
            }
            @Override
            public void onError(Throwable e) {
                Log.e(TAG, e.getMessage());
            }
            @Override
            public void onNext(Integer i) {
                Log.e(TAG, "onNext:"+i);
            }
        };
        observable.subscribe(subscriber);
        

輸出信息為

onNext:1
onNext:2
onNext:3
onNext:4
onCompleted

from操作符可以轉換Future、Iterable和數組。對于Iterable和數組,產生的Observable會發射Iterable或數組的每一項數據。對于Future,它會發射Future.get()方法返回的單個數據,并且還可以增加通過: from(Future,timeout, timeUnit)指定超時時間,如果執行的時候Future超時會回調onError()方法。

just

just將單個數據轉換為發射那個數據的Observable,Just類似于From,但是From會將數組或Iterable的數據取出然后逐個發射,而Just只是簡單的原樣發射,將數組或Iterable當做單個數據,如果你傳遞null給Just,它會返回一個發射null值的Observable。不要誤認為它會返回一個空Observable(完全不發射任何數據的Observable)。對于just可以接收1到10個數據,返回一個按參數列表順序發射這些數據的Observable。

        Observable.just(1 2, 3, 4)
                .subscribe(new Subscriber<Integer>() {
                    @Override
                    public void onCompleted() {
                        Log.e(TAG, "onCompleted: ");
                    }

                    @Override
                    public void onError(Throwable e) {
                        Log.e(TAG, "onError: ");
                    }

                    @Override
                    public void onNext(Integer integer) {
                        Log.e(TAG, "onNext: " + integer);
                    }
                });

輸出

onNext:1
onNext:2
onNext:3
onNext:4
onCompleted:

對于just參數類型可以是多種,如下,傳入兩個類型數據

Observable.just(0, "one", 6, "two", 8, "three")
                .subscribe(new Subscriber<Serializable>() {
                    @Override
                    public void onCompleted() {
                        Log.e(TAG, "onCompleted: " );
                    }

                    @Override
                    public void onError(Throwable e) {
                        Log.e(TAG, "onError: ");
                    }

                    @Override
                    public void onNext(Serializable serializable) {
                        Log.e(TAG, "onNext: "+serializable.toString());
                    }
                });

則輸出信息

onNext:0
onNext:one
onNext:6
onNext:two
onNext:8
onNext:three
onCompleted:

Empty/Never/Error

Empty:創建一個不發射任何數據但是正常終止的Observable,此時會回調onCompleted()

Never:創建一個不發射數據也不終止的Observable

Error:創建一個不發射數據以一個錯誤終止的Observable

error操作符需要一個Throwable參數,你的Observable會以此終止。這些操作符默認不在任何特定的調度器上執行,但是empty和error有一個可選參數是Scheduler,如果你傳遞了Scheduler參數,它們會在你指定的調度器上發送通知。

Range

該操作符創建特定整數序列的Observable,它接受兩個參數,一個是范圍的起始值,一個是范圍的數據的數目。如果你將第二個參數設為0,將導致Observable不發射任何數據(如果設置為負數,會拋異常)。

        Observable.range(1,4)
                .subscribe(new Subscriber<Integer>() {
                    public String TAG="RXJAVA";

                    @Override
                    public void onCompleted() {
                        Log.e(TAG, "onCompleted: " );
                    }

                    @Override
                    public void onError(Throwable e) {
                        Log.e(TAG, "onError: ");
                    }

                    @Override
                    public void onNext(Integer integer) {
                        Log.e(TAG, "onNext: "+integer);
                    }
                });

輸出信息

onNext: 1
onNext: 2
onNext: 3
onNext: 4
onCompleted: 

你可以在代碼實戰中,更改第二個參數為負數,或者0,以及將第一個參數更改為你想測試的任意值,去觀察執行日志幫助理解。

Timer

Timer操作符創建一個在給定的時間段之后返回一個特殊值的Observable。它在延遲一段給定的時間后發射一個簡單的數字0 。

Observable.timer(1, TimeUnit.SECONDS)
                .subscribe(new Subscriber<Long>() {
                    public String TAG="RXJAVA";

                    @Override
                    public void onCompleted() {
                        Log.e(TAG, "onCompleted: " );
                    }

                    @Override
                    public void onError(Throwable e) {
                        Log.e(TAG, "onError: ");
                    }

                    @Override
                    public void onNext(Long integer) {
                        Log.e(TAG, "onNext:1111111 "+integer);
                    }
                });

對于該操作符默認在computation調度器上執行的,如果你想在onNext()回調方法更新UI,需要通過observeOn(AndroidSchedulers.mainThread())設置,否則會調用onError()方法。當然Time人提供的有一個三個參數的方法timer(long,TimeUnit,Scheduler)可以指定 Scheduler 。

Interval

該操作符按固定的時間間隔發射一個無限遞增的整數序列,它接受一個表示時間間隔的參數和一個表示時間單位的參數,當然該操作符合Timer一樣,是在computation調度器上執行的,若想更新UI需要指定Scheduler 為AndroidSchedulers.mainThread()。

     Subscription   subscription = Observable.interval(1, TimeUnit.SECONDS)
                .observeOn(AndroidSchedulers.mainThread())
                .subscribe(new Action1<Long>() {
                    @Override
                    public void call(Long aLong) {
                        tv.append(" " + aLong + "   ");
                    }
                });

通過上面代碼就會每隔1秒在tv上追加一個數字,并且會永遠執行。如果在某個時刻不想繼續輸出,就需要要解除訂閱。

        if (subscription != null && !subscription.isUnsubscribed()) {
            subscription.unsubscribe();
        }

Repeat

該操作符是重復的發射某個數據序列,并且可以自己設置重復的次數。當接收到onComplete()會觸發重訂閱再次重復發射數據,當重復發射數據次數到達后執行onCompleted()。

        String[] strs = {"也許當初忙著微笑和哭泣", "忙著追逐天空中的流星"};
        Observable.from(strs).repeat(2)..subscribe(new Subscriber<String>() {
            @Override
            public void onCompleted() {
                Log.e(TAG, "onCompleted: " );
            }

            @Override
            public void onError(Throwable e) {
                Log.e(TAG, "onError: ");
            }

            @Override
            public void onNext(String s) {
                Log.e(TAG, "onNext: "+s );
                tv1.append("\n" + s);
            }
        });

輸出

onNext: 也許當初忙著微笑和哭泣
onNext: 忙著追逐天空中的流星
onNext: 也許當初忙著微笑和哭泣
onNext: 忙著追逐天空中的流星
onCompleted: 

Defer

直到有觀察者訂閱時才創建Observable,并且為每個觀察者創建一個新的Observable,該操作符能保證訂閱執行時數據源是最新的數據。如下正常代碼

        String test="舊數據";
        Observable observable=Observable.just(test);
        Subscriber subscriber=new Subscriber() {
            public String TAG="RXJAVA";

            @Override
            public void onCompleted() {
                Log.e(TAG, "onCompleted: " );
            }

            @Override
            public void onError(Throwable e) {
                Log.e(TAG, "onError: " );
            }

            @Override
            public void onNext(Object o) {
                Log.e(TAG, "onNext: "+o );
            }
        };
        test="新數據";
        observable.subscribe(subscriber);

輸出

 onNext: 舊數據
 onCompleted: 

通過上面代碼和輸出日志發現,雖然在后面講數據test更新為新數據,但是并沒有生效,要想使用最新的數據就需要使用defer操作符。此時更改使用defer

        test="舊數據";
        Observable<String> observable=Observable.defer(new Func0<Observable<String>>() {
            @Override
            public Observable<String> call() {
                return Observable.just(test);
            }
        });
        Subscriber subscriber=new Subscriber() {
            public String TAG="RXJAVA";

            @Override
            public void onCompleted() {
                Log.e(TAG, "onCompleted: " );
            }

            @Override
            public void onError(Throwable e) {
                Log.e(TAG, "onError: " );
            }

            @Override
            public void onNext(Object o) {
                Log.e(TAG, "onNext: "+o );
            }
        };
        test="新數據";
        observable.subscribe(subscriber);

輸出信息

onNext: 新數據
onCompleted: 

通過新的打印信息,發現輸出值已經是最新的數據。

到這里,這篇文章暫時就先結束了,若文章有不足或者錯誤的地方,歡迎指正,以防止給其他讀者錯誤引導。更多的操作符,將在接下來的幾篇文章介紹。

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

推薦閱讀更多精彩內容

  • 本篇文章介主要紹RxJava中操作符是以函數作為基本單位,與響應式編程作為結合使用的,對什么是操作、操作符都有哪些...
    嘎啦果安卓獸閱讀 2,887評論 0 10
  • 注:只包含標準包中的操作符,用于個人學習及備忘參考博客:http://blog.csdn.net/maplejaw...
    小白要超神閱讀 2,213評論 2 8
  • 作者: maplejaw本篇只解析標準包中的操作符。對于擴展包,由于使用率較低,如有需求,請讀者自行查閱文檔。 創...
    maplejaw_閱讀 45,753評論 8 93
  • 創建操作 用于創建Observable的操作符Create通過調用觀察者的方法從頭創建一個ObservableEm...
    rkua閱讀 1,854評論 0 1
  • 注:本系列文章主要用于博主個人學習記錄,本文末尾附上了一些較好的文章提供學習。轉載請附 原文鏈接RxJava學習系...
    黑丫山上小旋風閱讀 2,161評論 1 5