RxJava 1升級(jí)到RxJava 2過(guò)程中踩過(guò)的一些“坑”
RxJava2介紹
RxJava2 發(fā)布已經(jīng)有一段時(shí)間了,是對(duì) RxJava 的一次重大的升級(jí),由于我的一個(gè)庫(kù)cv4j使用了 RxJava2 來(lái)嘗鮮,但是 RxJava2 跟 RxJava1 是不能同時(shí)存在于一個(gè)項(xiàng)目中的,逼不得已我得把自己所有框架中使用 RxJava 的地方以及
App 中使用 RxJava 的地方都升級(jí)到最新版本。所以我整理并記錄了一些已經(jīng)填好的坑。分享出來(lái)供大家參考學(xué)習(xí),下面來(lái)看看詳細(xì)的介紹:
填坑記錄
1. RxJava1 跟 RxJava2 不能共存
如果,在同一個(gè)module中同時(shí)使用RxJava1和RxJava2,類(lèi)似如下:
compile 'io.reactivex.rxjava2:rxandroid:2.0.1' compile 'io.reactivex.rxjava2:rxjava:2.0.7' compile 'io.reactivex:rxandroid:1.2.0' compile 'io.reactivex:rxjava:1.1.5'
那么,很不幸你會(huì)遇到這樣的錯(cuò)誤
同理,在 App 中如果使用了 Rxjava2,但是某個(gè)第三方的 library 還在使用 Rxjava1 也會(huì)遇到同樣的錯(cuò)誤。
上面的錯(cuò)誤是因?yàn)?RxAndroid 2.0.1 本身依賴(lài)了 RxJava 2.0.1。我們嘗試去掉對(duì) RxJava 的依賴(lài),只留下 RxAndroid 。還是會(huì)遇到問(wèn)題。
compile 'io.reactivex.rxjava2:rxandroid:2.0.1' //compile 'io.reactivex.rxjava2:rxjava:2.0.7' compile 'io.reactivex:rxandroid:1.2.0' //compile 'io.reactivex:rxjava:1.1.5'
所以使用RxAndroid不能去掉對(duì)RxJava的依賴(lài),我是這樣使用的。
compile 'io.reactivex.rxjava2:rxandroid:2.0.1' compile 'io.reactivex.rxjava2:rxjava:2.0.7'
官方也是這樣解釋的
Because RxAndroid releases are few and far between, it is recommended you also explicitly depend on RxJava's latest version for bug fixes and new features.
最后,我建議要升級(jí)到 RxJava2 的時(shí)候必須所有使用的地方都要升級(jí),并且用最新的版本。
2. 新增Flowable
RxJava1 中 Observable 不能很好地支持 backpressure ,會(huì)拋出MissingBackpressureException。所以在 RxJava2 中 Oberservable 不再支持 backpressure ,而使用新增的 Flowable 來(lái)支持 backpressure 。
Flowable的用法跟原先的Observable是一樣的。
3. ActionN 和 FuncN 改名
ActionN 和 FuncN 遵循Java 8的命名規(guī)則。
其中,Action0 改名成Action,Action1改名成Consumer,而Action2改名成了BiConsumer,而Action3 - Action9都不再使用了,ActionN變成了Consumer<Object[]>
。
同樣,F(xiàn)unc改名成Function,F(xiàn)unc2改名成BiFunction,F(xiàn)unc3 - Func9 改名成 Function3 - Function9,F(xiàn)uncN 由 Function<Object[], R>
取代。
4. Observable.OnSubscribe 變成 ObservableOnSubscribe
原先RxJava1的寫(xiě)法:
Observable.create(new Observable.OnSubscribe<String>() { @Override public void call(Subscriber<? super String> subscriber) { subscriber.onNext("hello"); } }).subscribe(new Action1<String>() { @Override public void call(String s) { System.out.println(s); } });
現(xiàn)在的寫(xiě)法:
Observable.create(new ObservableOnSubscribe<String>() { @Override public void subscribe(ObservableEmitter<String> e) throws Exception { e.onNext("hello"); } }).subscribe(new Consumer<String>() { @Override public void accept(String s) { System.out.println(s); } });
5. ObservableOnSubscribe 中使用 ObservableEmitter 發(fā)送數(shù)據(jù)給 Observer
結(jié)合上一條,ObservableOnSubscribe 不再使用 Subscriber 而是用 ObservableEmitter 替代。
ObservableEmitter 可以理解為發(fā)射器,是用來(lái)發(fā)出事件的,它可以發(fā)出三種類(lèi)型的事件,通過(guò)調(diào)用emitter的onNext(T value)
、onComplete()
和onError(Throwable error)
可以分別發(fā)出next事件、complete事件和error事件。 如果只關(guān)心next事件的話,只需單獨(dú)使用onNext()
即可。
需要特別注意,emitter的onComplete()
調(diào)用后,Consumer不再接收任何next事件。
6. Observable.Transformer 變成 ObservableTransformer
原先RxJava1的寫(xiě)法:
/** * 跟compose()配合使用,比如ObservableUtils.wrap(obj).compose(toMain()) * @param <T> * @return */ public static <T> Observable.Transformer<T, T> toMain() { return new Observable.Transformer<T, T>() { @Override public Observable<T> call(Observable<T> tObservable) { return tObservable .subscribeOn(Schedulers.io()) .observeOn(AndroidSchedulers.mainThread()); } }; }
現(xiàn)在的寫(xiě)法:
/** * 跟compose()配合使用,比如ObservableUtils.wrap(obj).compose(toMain()) * @param <T> * @return */ public static <T> ObservableTransformer<T, T> toMain() { return new ObservableTransformer<T, T>() { @Override public ObservableSource<T> apply(Observable<T> upstream) { return upstream.subscribeOn(Schedulers.io()) .observeOn(AndroidSchedulers.mainThread()); } }; }
由于新增了Flowable,同理也增加了FlowableTransformer
public static <T> FlowableTransformer<T, T> toMain() { return new FlowableTransformer<T, T>() { @Override public Publisher<T> apply(Flowable<T> upstream) { return upstream.subscribeOn(Schedulers.io()) .observeOn(AndroidSchedulers.mainThread()); } }; }
7. Subscription 改名為 Disposable
在 RxJava2 中,由于已經(jīng)存在了 org.reactivestreams.subscription
這個(gè)類(lèi),為了避免名字沖突將原先的 rx.Subscription
改名為 io.reactivex.disposables.Disposable
。
剛開(kāi)始不知道,在升級(jí) RxJava2 時(shí)發(fā)現(xiàn) org.reactivestreams.subscription
這個(gè)類(lèi)完全沒(méi)法做原先 rx.Subscription
的事情:(
順便說(shuō)下,Disposable必須單次使用,用完就要銷(xiāo)毀。
8. first() 用法改變
官方文檔是這么描述的first()
的用法
1.x | 2.x |
---|---|
first() | RC3 renamed to firstElement and returns Maybe<T> |
first(Func1) | dropped, use filter(predicate).first() |
firstOrDefault(T) | renamed to first(T) and RC3 returns Single<T> |
firstOrDefault(Func1, T) | renamed to first(T) and RC3 returns Single<T> |
以first(Func1)
為例,first(Func1)
后面還使用了push()
,原先 Rxjava1會(huì)這樣寫(xiě)
ConnectableObservable<Data> connectableObservable = Observable .concat(Observable.from(list)) .first(new Func1<Data, Boolean>() { @Override public Boolean call(Data data) { return DataUtils.isAvailable(data); } }).publish();
RxJava2 改成這樣
ConnectableObservable<Data> connectableObservable = Observable .concat(Observable.fromIterable(list)) .filter(new Predicate<Data>() { @Override public boolean test(@NonNull Data data) throws Exception { return DataUtils.isAvailable(data); } }).firstElement().toObservable().publish();
9. toBlocking().y 被 blockingY() 取代
在我的框架中存在著一個(gè)Optional類(lèi),它跟Java 8的Optional作用差不多,原先是使用RxJava1來(lái)編寫(xiě)的。
import rx.Observable; /** * 使用方法: * String s = null; * Optional.ofNullable(s).orElse("default")); // 如果s為null,則顯示default,否則顯示s的值 * @author Tony Shen * */ public class Optional<T> { Observable<T> obs; public Optional(Observable<T> obs) { this.obs = obs; } public static <T> Optional<T> of(T value) { if (value == null) { throw new NullPointerException(); } else { return new Optional<T>(Observable.just(value)); } } public static <T> Optional<T> ofNullable(T value) { if (value == null) { return new Optional<T>(Observable.<T>empty()); } else { return new Optional<T>(Observable.just(value)); } } public T get() { return obs.toBlocking().single(); } public T orElse(T defaultValue) { return obs.defaultIfEmpty(defaultValue).toBlocking().single(); } }
升級(jí)到RxJava2之后,get()
和 orElse()
方法都會(huì)報(bào)錯(cuò),修改之后是這樣的。
import io.reactivex.Observable; /** * 使用方法: * String s = null; * Optional.ofNullable(s).orElse("default"); // 如果s為null,則顯示default,否則顯示s的值 * @author Tony Shen * */ public class Optional<T> { Observable<T> obs; public Optional(Observable<T> obs) { this.obs = obs; } public static <T> Optional<T> of(T value) { if (value == null) { throw new NullPointerException(); } else { return new Optional<T>(Observable.just(value)); } } public static <T> Optional<T> ofNullable(T value) { if (value == null) { return new Optional<T>(Observable.<T>empty()); } else { return new Optional<T>(Observable.just(value)); } } public T get() { return obs.blockingSingle(); } public T orElse(T defaultValue) { return obs.defaultIfEmpty(defaultValue).blockingSingle(); } }
10. PublishSubject
包括 PublishSubject 以及各種 Subject(ReplaySubject、BehaviorSubject、AsyncSubject) 都不再支持backpressure。
總結(jié)
RxJava2 所帶來(lái)的變化遠(yuǎn)遠(yuǎn)不止這些,以后遇到的話還會(huì)繼續(xù)整理和總結(jié),畢竟我使用的 RxJava2 還是很少的一部分內(nèi)容。
RxJava2 最好到文檔依然是官方文檔。如果是新項(xiàng)目到話,可以毫不猶豫地使用RxJava2,如果是在線上已經(jīng)成熟穩(wěn)定的項(xiàng)目,可以再等等。對(duì)于新手的話,可以直接從 RxJava2 學(xué)起,RxJava1 就直接略過(guò)吧。對(duì)于老手,RxJava2 還是使用原來(lái)的思想,區(qū)別不大,從 RxJava1 遷移到 Rxjava2 也花不了多少工夫。
好了,以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作能帶來(lái)一定的幫助,如果有疑問(wèn)大家可以留言交流,謝謝大家對(duì)腳本之家的支持。
- Android性能優(yōu)化之利用Rxlifecycle解決RxJava內(nèi)存泄漏詳解
- RxJava入門(mén)之介紹與基本運(yùn)用
- RxJava入門(mén)指南及其在Android開(kāi)發(fā)中的使用示例
- Android中的Retrofit+OkHttp+RxJava緩存架構(gòu)使用
- Android上傳多張圖片的實(shí)例代碼(RxJava異步分發(fā))
- RxJava2.x實(shí)現(xiàn)定時(shí)器的實(shí)例代碼
- 簡(jiǎn)單談?wù)凴xJava和多線程并發(fā)
- Android 使用 RxJava2 實(shí)現(xiàn)倒計(jì)時(shí)功能的示例代碼
- RxJava+Retrofit+OkHttp實(shí)現(xiàn)多文件下載之?dāng)帱c(diǎn)續(xù)傳
- 你用不慣 RxJava,只因缺了這把鑰匙(推薦)
相關(guān)文章
Android學(xué)習(xí)小結(jié)之Activity保存和恢復(fù)狀態(tài)
這篇文章主要介紹了Activity狀態(tài)保存和恢復(fù)的相關(guān)資料,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2016-08-08Flutter本地存儲(chǔ)之基本的鍵值對(duì)存儲(chǔ)詳解
在原生的?Android?或?iOS?中,都提供了基本的鍵值對(duì)存儲(chǔ)方式,在?Flutter?中,提供了?shared_preferences?這個(gè)插件來(lái)實(shí)現(xiàn)本地鍵值對(duì)數(shù)據(jù)存儲(chǔ),本文就來(lái)和大家簡(jiǎn)單聊聊吧2023-03-03Android空心圓及層疊效果實(shí)現(xiàn)代碼
這篇文章主要為大家詳細(xì)介紹了Android空心圓及層疊效果實(shí)現(xiàn)代碼,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-10-10Android GPS室內(nèi)定位問(wèn)題的解決方法(location為null)
這篇文章主要為大家詳細(xì)介紹了Android GPS室內(nèi)定位問(wèn)題的解決方法,location為null,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-02-02Android BottomNavigationBar底部導(dǎo)航的使用方法
這篇文章主要為大家詳細(xì)介紹了Android BottomNavigationBar底部導(dǎo)航的使用方法,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-11-11Android Universal ImageLoader 緩存圖片
Universal Image Loader for Android的目的是為了實(shí)現(xiàn)異步的網(wǎng)絡(luò)圖片加載、緩存及顯示,支持多線程異步加載,通過(guò)本文給大家介紹Android Universal ImageLoader緩存圖片相關(guān)資料,涉及到imageloader緩存圖片相關(guān)知識(shí),對(duì)imageloader緩存圖片相關(guān)知識(shí)感興趣的朋友一起學(xué)習(xí)2016-01-01Android NestedScrolling嵌套滾動(dòng)的示例代碼
這篇文章主要介紹了Android NestedScrolling嵌套滾動(dòng)的示例代碼,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-05-05