RxJava2 實戰(zhàn)系列文章
RxJava2 實戰(zhàn)知識梳理(1) - 后臺執(zhí)行耗時操作,實時通知 UI 更新
RxJava2 實戰(zhàn)知識梳理(2) - 計算一段時間內(nèi)數(shù)據(jù)的平均值
RxJava2 實戰(zhàn)知識梳理(3) - 優(yōu)化搜索聯(lián)想功能
RxJava2 實戰(zhàn)知識梳理(4) - 結(jié)合 Retrofit 請求新聞資訊
RxJava2 實戰(zhàn)知識梳理(5) - 簡單及進(jìn)階的輪詢操作
RxJava2 實戰(zhàn)知識梳理(6) - 基于錯誤類型的重試請求
RxJava2 實戰(zhàn)知識梳理(7) - 基于 combineLatest 實現(xiàn)的輸入表單驗證
RxJava2 實戰(zhàn)知識梳理(8) - 使用 publish + merge 優(yōu)化先加載緩存,再讀取網(wǎng)絡(luò)數(shù)據(jù)的請求過程
RxJava2 實戰(zhàn)知識梳理(9) - 使用 timer/interval/delay 實現(xiàn)任務(wù)調(diào)度
RxJava2 實戰(zhàn)知識梳理(10) - 屏幕旋轉(zhuǎn)導(dǎo)致 Activity 重建時恢復(fù)任務(wù)
RxJava2 實戰(zhàn)知識梳理(11) - 檢測網(wǎng)絡(luò)狀態(tài)并自動重試請求
RxJava2 實戰(zhàn)知識梳理(12) - 實戰(zhàn)講解 publish & replay & share & refCount & autoConnect
RxJava2 實戰(zhàn)知識梳理(13) - 如何使得錯誤發(fā)生時不自動停止訂閱關(guān)系
RxJava2 實戰(zhàn)知識梳理(14) - 在 token 過期時,刷新過期 token 并重新發(fā)起請求
RxJava2 實戰(zhàn)知識梳理(15) - 實現(xiàn)一個簡單的 MVP + RxJava + Retrofit 應(yīng)用
一、前言
其實在之前的文章中,我們已經(jīng)接觸過定時/周期執(zhí)行任務(wù)的操作符,例如在 RxJava2 實戰(zhàn)知識梳理(5) - 簡單及進(jìn)階的輪詢操作 和 RxJava2 實戰(zhàn)知識梳理(6) - 基于錯誤類型的重試請求 這兩篇文章當(dāng)中,我們通過intervalRange
實現(xiàn)了輪詢工作,通過timer
實現(xiàn)了延時的重試請求。
今天這篇文章,我們根據(jù) RxJava-Android-Samples 中總結(jié)的幾種場景來復(fù)習(xí)一下可以實現(xiàn)定時調(diào)度任務(wù)的操作符:
-
timer
:創(chuàng)建型操作符,用于延時執(zhí)行任務(wù)。 -
interval
:創(chuàng)建型操作符,用于周期執(zhí)行任務(wù)。 -
delay
:輔助型操作,用于延時傳遞數(shù)據(jù)。
二、timer
2.1 timer 操作符原理
timer
原理圖如下所示:
很簡單,在訂閱之后,它會在等待一段時間之后發(fā)射一個
0
數(shù)據(jù)項,然后結(jié)束,因此它常常可以用來延時地發(fā)送時間,例如 RxJava2 實戰(zhàn)知識梳理(5) - 簡單及進(jìn)階的輪詢操作 中,我們使用repeatWhen
發(fā)起重訂閱時,就是通過timer
實現(xiàn)了延時發(fā)送onNext
時間來實現(xiàn)時延變長的輪詢操作。
2.2 使用 timer 延時 1s 后執(zhí)行任務(wù),然后結(jié)束
我們使用timer
操作符實現(xiàn)下面的效果:延時1s
后在子線程執(zhí)行任務(wù),接著完成,這里我們采用了timer
,并通過subscribe
方法讓下游運(yùn)行在子線程當(dāng)中。
//延遲 1s 后執(zhí)行一個任務(wù),然后結(jié)束
private void startTimeDemo1() {
Log.d(TAG, "startTimeDemo1");
DisposableObserver<Long> disposableObserver = getTimeDemoObserver();
Observable.timer(1000, TimeUnit.MILLISECONDS).subscribeOn(Schedulers.io()).subscribe(disposableObserver);
mCompositeDisposable.add(disposableObserver);
}
運(yùn)行結(jié)果為:
三、interval
3.1 interval 原理
interval
的原理圖如下所示:
interval
也是一個創(chuàng)建型操作符,它可以間隔一段時間就發(fā)送一個數(shù)據(jù)。
3.2 每隔 1s 執(zhí)行一次任務(wù),第一次任務(wù)執(zhí)行前有 1s 的間隔,執(zhí)行無限次
我們先使用interval
實現(xiàn)下面這個效果:每隔1s
執(zhí)行一次任務(wù),第一次任務(wù)執(zhí)行前有1s
的間隔,執(zhí)行無限次。這是因為,使用interval
操作符時,默認(rèn)第一次個任務(wù)需要延時和指定間隔相同的時間。
//每隔 1s 執(zhí)行一次任務(wù),第一次任務(wù)執(zhí)行前有 1s 的間隔,執(zhí)行無限次
private void startTimeDemo2() {
Log.d(TAG, "startTimeDemo2");
DisposableObserver<Long> disposableObserver = getTimeDemoObserver();
Observable.interval(1000, TimeUnit.MILLISECONDS).subscribeOn(Schedulers.io()).subscribe(disposableObserver);
mCompositeDisposable.add(disposableObserver);
}
運(yùn)行結(jié)果如下所示:
3.3 每隔 1s 執(zhí)行一次任務(wù),立即執(zhí)行第一次任務(wù),執(zhí)行無限次
如果希望立即執(zhí)行第一次任務(wù),那么可以給它提供額外的參數(shù),指定第一次任務(wù)的延時:
//每隔 1s 執(zhí)行一次任務(wù),立即執(zhí)行第一次任務(wù),執(zhí)行無限次
private void startTimeDemo3() {
Log.d(TAG, "startTimeDemo3");
DisposableObserver<Long> disposableObserver = getTimeDemoObserver();
Observable.interval(0, 1000, TimeUnit.MILLISECONDS).subscribeOn(Schedulers.io()).subscribe(disposableObserver);
mCompositeDisposable.add(disposableObserver);
}
運(yùn)行結(jié)果為:
3.4 每隔 1s 執(zhí)行一次任務(wù),立即執(zhí)行第一次任務(wù),只執(zhí)行五次
在3.3
的例子中,我們的任務(wù)會無限執(zhí)行下去,如果我們希望只執(zhí)行指定次數(shù)該怎么辦呢,其實在 RxJava2 實戰(zhàn)知識梳理(5) - 簡單及進(jìn)階的輪詢操作 中演示固定時延的輪詢操作時,我們已經(jīng)介紹了使用intervalRange
來實現(xiàn),今天,我們采用interval + take
的方式來實現(xiàn),代碼如下:
//每隔 1s 執(zhí)行一次任務(wù),立即執(zhí)行第一次任務(wù),只執(zhí)行五次
private void startTimeDemo4() {
Log.d(TAG, "startTimeDemo4");
DisposableObserver<Long> disposableObserver = getTimeDemoObserver();
Observable.interval(0, 1000, TimeUnit.MILLISECONDS).take(5).subscribe(disposableObserver);
mCompositeDisposable.add(disposableObserver);
}
運(yùn)行結(jié)果為:
take
的原理圖如下所示:它表示我們只接受前
n
個數(shù)據(jù)項,這樣和interval
結(jié)合就可以實現(xiàn)固定間隔與固定次數(shù)的任務(wù)執(zhí)行。
四、delay
4.1 delay 原理
delay
的原理圖如下所示:
當(dāng)它接受一個時間段時,每當(dāng)原始的
Observable
發(fā)射了一個數(shù)據(jù)項時,它就啟動一個定時器,等待指定的時間后再將這個數(shù)據(jù)發(fā)射出去,因此表現(xiàn)為發(fā)射的數(shù)據(jù)項進(jìn)行了平移,但是它只會平移onNext/onComplete
,對于onError
,它會立即發(fā)射出去,并且丟棄之前等待發(fā)射的onNext
事件。
4.2 先執(zhí)行一個任務(wù),等待 1s,再執(zhí)行另一個任務(wù),然后結(jié)束
因為delay
不是創(chuàng)建型操作符,所以我們可以用來延遲上游發(fā)射過來的數(shù)據(jù),下面,讓我們實現(xiàn)這個效果:先執(zhí)行一個任務(wù),等待 1s,再執(zhí)行另一個任務(wù),然后結(jié)束。代碼如下:
//先執(zhí)行一個任務(wù),等待 1s,再執(zhí)行另一個任務(wù),然后結(jié)束
private void startTimeDemo5() {
Log.d(TAG, "startTimeDemo5");
DisposableObserver<Long> disposableObserver = getTimeDemoObserver();
Observable.just(0L).doOnNext(new Consumer<Long>() {
@Override
public void accept(Long aLong) throws Exception {
Log.d(TAG, "執(zhí)行第一個任務(wù)");
}
}).delay(1000, TimeUnit.MILLISECONDS).subscribe(disposableObserver);
mCompositeDisposable.add(disposableObserver);
}
執(zhí)行效果為:
更多文章,歡迎訪問我的 Android 知識梳理系列:
- Android 知識梳理目錄:http://www.lxweimin.com/p/fd82d18994ce
- 個人主頁:http://lizejun.cn
- 個人知識總結(jié)目錄:http://lizejun.cn/categories/