前言
通過本系列上一篇文章,我們了解了RxJava1中由于被觀察者發送事件的速度遠快于觀察者處理事件的速度導致的背壓問題,由此而生出背壓策略。雖然RxJava1中有一些操作符支持背壓策略,也有一些其他方法,但效果并不太理想。這個問題到了RxJava2中得到了更好地解決。本篇一起來看相較于RxJava1,RxJava2從觀察者模式到語法以及線程調度都做了哪些更新。
依賴
首先當然是添加RxJava2與RxAndroid2的依賴啦,以當前最新版本為準。雖然RxAndroid中包含RxJava的內容,但嘗試了一下,只引入RxAndroid還是會報錯,所以兩依賴都加上了。
dependencies {
......
compile 'io.reactivex.rxjava2:rxjava:2.1.3'
compile 'io.reactivex.rxjava2:rxandroid:2.0.1'
}
注意:RxJava2與RxJava1的依賴不可共存
觀察者模式更新
RxJava2中除了保留了RxJava1中Observable-->Observer觀察者模式以外,還新增了一種觀察者模式Flowable-->Subscriber。目的為何相信大家已經知道了,新觀察者模式可完美支持背壓策略。
不了解背壓策略的碼友歡迎閱讀上一篇文章:
RxJava2系列之背壓策略(一)
由結構圖可見,RxJava2將RxJava1中支持背壓與不支持背壓的操作符分開了,支持背壓的歸入Flowable觀察者模式中,不支持背壓的歸入Observable觀察者模式中。
來看一個使用range操作符的Flowable示例:
Flowable.range(1, 4).subscribe(new Subscriber<Integer>() {
Subscription subscription;
@Override
public void onSubscribe(Subscription s) {
Log.e("rx2_test", "onSubscribe");
subscription = s;
//這里可進行一些初始化操作
subscription.request(1);
}
@Override
public void onNext(Integer integer) {
Log.e("rx2_test", "onNext:" + integer);
subscription.request(1);
}
@Override
public void onError(Throwable t) {
Log.e("rx2_test", "onError:" + t.getMessage());
}
@Override
public void onComplete() {
Log.e("rx2_test", "onComplete");
}
});
輸出結果:
rx2_test: onSubscribe
rx2_test: onNext:1
rx2_test: onNext:2
rx2_test: onNext:3
rx2_test: onNext:4
rx2_test: onComplete
由示例可以看出Flowable是支持背壓策略的,整個數據流是由Subscriber動態去請求,然后Flowable再響應發射事件。訂閱之后首先回調onSubscribe()方法,相當于onStart()方法,建議在其中先進行一些初始化的操作然后再調用request(n)請求數據。示例中的Subscription可用于請求數據或在發射過程中滿足某些條件時進行取消訂閱的操作。
再來看一個使用create操作符的Flowable示例:
Flowable.create((FlowableOnSubscribe<Integer>) e -> {
e.onNext(1);
e.onNext(2);
e.onNext(3);
e.onNext(4);
e.onComplete();
},
BackpressureStrategy.BUFFER)
.subscribe(new MySubscription());
輸出結果:
rx2_test: onSubscribe
rx2_test: onNext:1
rx2_test: onNext:2
rx2_test: onNext:3
rx2_test: onNext:4
rx2_test: onComplete
Flowable也可使用create操作符來創建,但需指定背壓策略BackpressureStrategy.BUFFER
。
除此之外還有幾種觀察者模式:
- Single/SingleObserver:訂閱后只能接收到一次
- Completable/CompletableObserver:只能接收到完成(onComplete)和錯誤(onError)回調。
- Maybe/MaybeObserver:只能接收到一次true或false的數據
這幾種觀察者模式主要用來判斷單一狀態,不用來發送大量數據。實際使用頻率不高就不一一贅述了。
語法更新
語法更新方面,大部分操作符的用法都未改動,有改動的是RxJava1中ActionN,FuncN等等的包名類名。
Null
經落木語碼友的提醒,首先需要注意一個細節變動:RxJava2中不再支持傳入null,否則會拋出NullPointerException。
RxJava1中:
//可直接傳入null
Observable.just(null);
RxJava2中:
//直接傳入null會拋出NullPointerException
Observable.just(null);
//操作符返回null會走onError()回調
Observable.just(1)
.map(integer -> null)
.subscribe(o -> Log.e("test_rxjava", "onNext:" + o.toString()),
throwable -> Log.e("test_rxjava", "onError:" + throwable.getMessage()));
Action更新
更新內容見圖:
用法示例:
Flowable.just(1, 2, 3, 4, 5)
.subscribe(new Consumer<Integer>() {
@Override
public void accept(Integer integer) throws Exception {
Log.e("rx2_test", "actionChange:" + integer);
}
});
Func更新
更新內容見圖:
用法示例:
Flowable.just(1, 2, 3, 4, 5)
.map(new Function<Integer, String>() {
@Override
public String apply(@NonNull Integer integer) throws Exception {
return "xulei" + integer;
}
})
.subscribe(new Consumer<String>() {
@Override
public void accept(String s) throws Exception {
Log.e("rx2_test", "funcChange:" + s);
}
});
注意:在Rxjava2中,Action與Function的回調方法中都默認throws Exception,可以不進行try-catch的操作。
Subscription更新
Subscription改名為Disposable。由于RxJava2中已經存在了org.reactivestreams.subscription
這個類,為了避免名字沖突將原先的rx.Subscription
改名為io.reactivex.disposables.Disposable
。
CompositeSubscription也改名為CompositeDisposable。
注意:Disposable必須單次使用,用完就要銷毀。
需要強調的是,當一個Activity中有多個RxJava異步事件流也就是Disposable時,需要根據事務周期及時進行銷毀,否則會導致內存慢慢的,慢慢的......Bang!?。∧愣谩D窃趺催M行集體銷毀呢?一起來看下RxJava2中如何實現。
protected CompositeDisposable mCompositeDisposable;
/**
* 添加RxJava訂閱
*/
protected void addDisposable(Disposable disposable) {
if (mCompositeDisposable == null) {
mCompositeDisposable = new CompositeDisposable();
}
mCompositeDisposable.add(subscription);
}
/**
* 取消RxJava訂閱
*/
protected void clearDisposable() {
if (mCompositeDisposable != null) {
mCompositeDisposable.clear();
}
}
創建一個CompositeDisposable對象(替代RxJava1中CompositeSubscription),在實例化RxJava2中的Disposable時可調用其add(Disposable d)
將Disposable加入其中。等到Activity銷毀或者不需要時便可調用其clear()
方法對其中的Disposable進行統一取消訂閱并銷毀的操作。
使用示例:
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(......);
addDisposable(Flowable.just(1, 2, 3, 4, 5)
.subscribe((integer -> Log.e("rx2_test", "" + integer))));
}
@Override
protected void onDestroy() {
super.onDestroy();
clearDisposable()
}
建議:addSubscribe(Disposable disposable)
與clearDisposable()
可放入BaseActivity或者MVP架構中的BasePresenter中,便于統一管理,使代碼更加簡潔。
新增doOnCancel等
RxJava2中除了保留了RxJava1中doOnComplete()
,doOnError()
,doOnNext()
等狀態回調之外,還增加了doOnCancel
,doOnDispose
,unsubscribeOn
這幾個狀態回調,以doOnCancel
為例:
Flowable.range(1, 4)
.doOnCancel(new Action() {
@Override
public void run() throws Exception {
Log.e("rx2_test", "回調doOnCancel");
}
})
.take(2)
.subscribe(new Consumer<Integer>() {
@Override
public void accept(Integer integer) throws Exception {
Log.e("rx2_test", "doOnCancel:" + integer);
}
});
輸出結果:
rx2_test: doOnCancel:1
rx2_test: doOnCancel:2
rx2_test: 回調doOnCancel
由于.take(2)
操作符只截取了前兩個事件,之后的事件未被發送,就會回調doOnCancel中的run()方法。
線程調度更新
線程調度這里改動也不大,主要有以下三點:
- 移除Schedulers.immediate。
- 移除Schedulers.test。
- io.reactivex.Scheduler抽象類支持直接調度自定義線程任務。
總結
本文主要講解了RxJava2中常用點的更新,更多細節的調整各位碼友可以參考官網文檔。了解了理論還需勤加練習才能真正掌握RxJava2,下一篇我們將通過一個實際例子來實踐一下RxJava2,敬請期待。
進階中的碼猿一枚,寫的不對的地方歡迎大神們留言指正,有什么疑惑或者建議也可以在我Github上RxJava2Demo項目Issues中提出,我會及時回復。
附上RxJava2Demo的地址:
RxJava2Demo
注意:本文中插入的圖片均為本人原創繪制,引用請標注來源,謝謝。