300字范文,内容丰富有趣,生活中的好帮手!
300字范文 > JAVA RxJava 2.x

JAVA RxJava 2.x

时间:2021-12-23 13:14:13

相关推荐

JAVA RxJava 2.x

【JAVA RxJava 2.x】

为什么要学 RxJava?

RxJava 最大的优点也是简洁,但它不止是简洁,而且是* 随着程序逻辑变得越来越复杂,它依然能够保持简洁 *。

咳咳,要例子,猛戳这里:给 Android 开发者的 RxJava 详解

什么是响应式编程

响应式编程是一种基于异步数据流概念的编程模式。

数据流就像一条河:它可以被观测,被过滤,被操作,或者为新的消费者与另外一条流合并为一条新的流。

响应式编程的一个关键概念事件。事件可以被等待,可以触发过程,也可以触发其它事件。事件是唯一的以合适的方式将我们的现实世界映射到我们的软件中:如果屋里太热了我们就打开一扇窗户。同样的,当我们的天气app从服务端获取到新的天气数据后,我们需要更新app上展示天气信息的UI;汽车上的车道偏移系统探测到车辆偏移了正常路线就会提醒驾驶者纠正,就是是响应事件。

响应式编程最通用的一个场景是UI:我们的移动App必须做出对网络调用、用户触摸输入和系统弹框的响应。

简介

RxJava 2.x 已经按照 Reactive-Streams specification 规范完全的重写了,maven也被放在了io.reactivex.rxjava2:rxjava:2.x.y下,所以 RxJava 2.x 独立于 RxJava 1.x 而存在,而随后官方宣布的将在一段时间后终止对 RxJava 1.x 的维护,所以对于熟悉 RxJava 1.x 的老司机自然可以直接看一下 2.x 的文档和异同就能轻松上手了。

而对于不熟悉的年轻司机,此处有案例:/nanchen2251/RxJava2Examples

你只需要在 build.gradle 中加上:compile 'io.reactivex.rxjava2:rxjava:2.1.1'(2.1.1为写此文章时的最新版本)

接口变化

RxJava 2.x 拥有了新的特性,其依赖于4个基础接口,它们分别是

PublisherSubscriberSubscriptionProcessor

其中最核心的莫过于PublisherSubscriber

Publisher可以发出一系列的事件,而Subscriber负责和处理这些事件

其中用的比较多的自然是PublisherFlowable,它支持背压。关于背压给个简洁的定义就是:

背压是指在异步场景中,被观察者发送事件速度远快于观察者的处理速度的情况下,一种告诉上游的被观察者降低发送速度的策略。

简而言之,背压是流速控制的一种策略。有兴趣的可以看一下官方对于背压的讲解。

可以明显地发现,RxJava 2.x 最大的改动就是对于backpressure的处理,为此将原来的Observable拆分成了新的ObservableFlowable,同时其他相关部分也同时进行了拆分。

观察者模式

RxJava 以观察者模式为骨架,在 2.0 中依旧如此

不过此次更新中,出现了两种观察者模式:

Observable(被观察者 ) /Observer( 观察者 )Flowable被观察者)/Subscriber(观察者)

在 RxJava 2.x 中,Observable 用于订阅 Observer,不再支持背压(1.x 中可以使用背压策略),而 Flowable 用于订阅 Subscriber , 是支持背压(Backpressure)的。

Observable

RxJava 三部曲

第一步:初始化 Observable

第二步:初始化 Observer

第三步:建立订阅关系

Observable.create(new ObservableOnSubscribe<Integer>() { // 第一步:初始化Observable@Overridepublic void subscribe(@NonNull ObservableEmitter<Integer> e) throws Exception {Log.e(TAG, "Observable emit 1" + "\n");e.onNext(1);Log.e(TAG, "Observable emit 2" + "\n");e.onNext(2);Log.e(TAG, "Observable emit 3" + "\n");e.onNext(3);e.onComplete();Log.e(TAG, "Observable emit 4" + "\n" );e.onNext(4);}}).subscribe(new Observer<Integer>() { // 第三步:订阅// 第二步:初始化Observerprivate int I;private Disposable mDisposable;@Overridepublic void onSubscribe(@NonNull Disposable d) {mDisposable = d;}@Overridepublic void onNext(@NonNull Integer integer) {I++;if (i == 2) {// 在RxJava 2.x 中,新增的Disposable可以做到切断的操作,让Observer观察者不再接收上游事件mDisposable.dispose();}}@Overridepublic void onError(@NonNull Throwable e) {Log.e(TAG, "onError : value : " + e.getMessage() + "\n" );}@Overridepublic void onComplete() {Log.e(TAG, "onComplete" + "\n" );}});

与 1.x区别

RxJava 2.x 与 1.x 还是存在着一些区别的:

- 首先,创建 Observable时,回调的是ObservableEmitter,字面意思即发射器,并且直接 throws Exception。

- 其次,在创建的 Observer中,也了一个回调方法:onSubscribe,传递参数为DisposableDisposable相当于 RxJava 1.x 中的Subscription, 用于解除订阅。可以看到示例代码中,在 i 自增到 2 的时候,订阅关系被切断。

07-03 14:24:11.663 18467-18467/com.nanchen.rxjava2examples E/RxCreateActivity: onSubscribe : false07-03 14:24:11.664 18467-18467/com.nanchen.rxjava2examples E/RxCreateActivity: Observable emit 107-03 14:24:11.665 18467-18467/com.nanchen.rxjava2examples E/RxCreateActivity: onNext : value : 107-03 14:24:11.666 18467-18467/com.nanchen.rxjava2examples E/RxCreateActivity: Observable emit 207-03 14:24:11.667 18467-18467/com.nanchen.rxjava2examples E/RxCreateActivity: onNext : value : 207-03 14:24:11.668 18467-18467/com.nanchen.rxjava2examples E/RxCreateActivity: onNext : isDisposable : true07-03 14:24:11.669 18467-18467/com.nanchen.rxjava2examples E/RxCreateActivity: Observable emit 307-03 14:24:11.670 18467-18467/com.nanchen.rxjava2examples E/RxCreateActivity: Observable emit 4

简化订阅

当然,我们的 RxJava 2.x 也为我们保留了简化订阅方法,我们可以根据需求,进行相应的简化订阅,只不过传入对象改为了Consumer

Consumer即消费者,用于接收单个值,

BiConsumer则是接收两个值,

Function用于变换对象,

Predicate用于判断。

这些接口命名大多参照了 Java 8。

线程调度

关于线程切换这点,RxJava 1.x 和 RxJava 2.x 的实现思路是一样的。

subscribeOn

同 RxJava 1.x 一样,subscribeOn用于指定subscribe()时所发生的线程,从源码角度可以看出,内部线程调度是通过ObservableSubscribeOn来实现的。

@SchedulerSupport(SchedulerSupport.CUSTOM)public final Observable<T> subscribeOn(Scheduler scheduler) {ObjectHelper.requireNonNull(scheduler, "scheduler is null");return RxJavaPlugins.onAssembly(new ObservableSubscribeOn<T>(this, scheduler));}

ObservableSubscribeOn的核心源码在subscribeActual方法中,通过代理的方式使用SubscribeOnObserver包装Observer后,设置Disposable来将subscribe切换到Scheduler线程中。

observeOn

observeOn方法用于指定下游Observer回调发生的线程。

@SchedulerSupport(SchedulerSupport.CUSTOM)public final Observable<T> observeOn(Scheduler scheduler, boolean delayError, int bufferSize) {ObjectHelper.requireNonNull(scheduler, "scheduler is null");ObjectHelper.verifyPositive(bufferSize, "bufferSize");return RxJavaPlugins.onAssembly(new ObservableObserveOn<T>(this, scheduler, delayError, bufferSize));}

线程切换注意

RxJava 内置的线程调度器的确可以让我们的线程切换得心应手,但其中也有些需要注意的地方。

简单地说,subscribeOn()指定的就是发射事件的线程,observerOn指定的就是订阅者接收事件的线程。多次指定发射事件的线程只有第一次指定的有效,也就是说多次调用subscribeOn()只有第一次的有效,其余的会被忽略。但多次指定订阅者接收线程是可以的,也就是说每调用一次observerOn()下游的线程就会切换一次

Observable.create(new ObservableOnSubscribe<Integer>() {@Overridepublic void subscribe(@NonNull ObservableEmitter<Integer> e) throws Exception {Log.e(TAG, "Observable thread is : " + Thread.currentThread().getName());e.onNext(1);e.onComplete();}}).subscribeOn(Schedulers.newThread()).subscribeOn(Schedulers.io()).observeOn(AndroidSchedulers.mainThread()).doOnNext(new Consumer<Integer>() {@Overridepublic void accept(@NonNull Integer integer) throws Exception {Log.e(TAG, "After observeOn(mainThread),Current thread is " + Thread.currentThread().getName());}}).observeOn(Schedulers.io()).subscribe(new Consumer<Integer>() {@Overridepublic void accept(@NonNull Integer integer) throws Exception {Log.e(TAG, "After observeOn(io),Current thread is " + Thread.currentThread().getName());}});

输出:

07-03 14:54:01.177 15121-15438/com.nanchen.rxjava2examples E/RxThreadActivity: Observable thread is : RxNewThreadScheduler-107-03 14:54:01.178 15121-15121/com.nanchen.rxjava2examples E/RxThreadActivity: After observeOn(mainThread),Current thread is main07-03 14:54:01.179 15121-15439/com.nanchen.rxjava2examples E/RxThreadActivity: After observeOn(io),Current thread is RxCachedThreadScheduler-2

实例代码中,分别用 Schedulers.newThread() 和 Schedulers.io() 对发射线程进行切换,并采用 observeOn(AndroidSchedulers.mainThread() 和 Schedulers.io() 进行了接收线程的切换。可以看到输出中发射线程仅仅响应了第一个 newThread,但每调用一次 observeOn() ,线程便会切换一次,因此如果我们有类似的需求时,便知道如何处理了。

内置线程

RxJava 中,已经内置了很多线程选项供我们选择,例如有:

Schedulers.io()代表io操作的线程, 通常用于网络,读写文件等io密集型的操作;putation()代表CPU计算密集型的操作, 例如需要大量计算的操作;Schedulers.newThread()代表一个常规的新线程;AndroidSchedulers.mainThread()代表Android的主线程

这些内置的Scheduler已经足够满足我们开发的需求,因此我们应该使用内置的这些选项,而RxJava 内部使用的是线程池维护这些线程,所以效率也比较高。

操作符

关于操作符,在官方文档中已经做了非常完善的讲解,并且笔者前面的系列教程中也着重讲解了绝大多数的操作符作用,这里受于篇幅限制,就不多做赘述,只挑选几个进行实际情景的讲解。

map

map 操作符可以将一个 Observable 对象通过某种关系转换为另一个Observable 对象。

在 2.x 中和 1.x 中作用几乎一致,不同点在于:2.x将 1.x 中的Func1Func2改为了FunctionBiFunction

采用 map 操作符进行网络数据解析

想必大家都知道,很多时候我们在使用 RxJava 的时候总是和 Retrofit 进行结合使用,而为了方便演示,这里我们就暂且采用 OkHttp3 进行演示,配合 map,doOnNext ,线程切换进行简单的网络请求:

1)通过 Observable.create() 方法,调用 OkHttp 网络请求;

2)通过 map 操作符集合 gson,将 Response 转换为 bean 类;

3)通过 doOnNext() 方法,解析 bean 中的数据,并进行数据库存储等操作;

4)调度线程,在子线程中进行耗时操作任务,在主线程中更新 UI ;

5)通过 subscribe(),根据请求成功或者失败来更新 UI 。

Observable.create(new ObservableOnSubscribe<Response>() {@Overridepublic void subscribe(@NonNull ObservableEmitter<Response> e) throws Exception {Builder builder = new Builder().url("/MobilePlace/LookUp?key=ec47b85086be4dc8b5d941f5abd37a4e&mobileNumber=13021671512").get();Request request = builder.build();Call call = new OkHttpClient().newCall(request);Response response = call.execute();e.onNext(response);}}).map(new Function<Response, MobileAddress>() {@Overridepublic MobileAddress apply(@NonNull Response response) throws Exception {if (response.isSuccessful()) {ResponseBody body = response.body();if (body != null) {Log.e(TAG, "map:转换前:" + response.body());return new Gson().fromJson(body.string(), MobileAddress.class);}}return null;}}).observeOn(AndroidSchedulers.mainThread()).doOnNext(new Consumer<MobileAddress>() {@Overridepublic void accept(@NonNull MobileAddress s) throws Exception {Log.e(TAG, "doOnNext: 保存成功:" + s.toString() + "\n");}}).subscribeOn(Schedulers.io()).observeOn(AndroidSchedulers.mainThread()).subscribe(new Consumer<MobileAddress>() {@Overridepublic void accept(@NonNull MobileAddress data) throws Exception {Log.e(TAG, "成功:" + data.toString() + "\n");},new Consumer<Throwable>(){@Overridepublic void accept (@NonNull Throwable throwable) throws Exception {Log.e(TAG, "失败:" + throwable.getMessage() + "\n");}});

concat

concat 可以做到不交错的发射两个甚至多个 Observable 的发射事件,并且只有前一个 Observable 终止(onComplete) 后才会订阅下一个 Observable

采用 concat 操作符先读取缓存再通过网络请求获取数据

在实际应用中,很多时候(对数据操作不敏感时)都需要我们先读取缓存的数据,如果缓存没有数据,再通过网络请求获取,随后在主线程更新我们的UI。

concat 操作符简直就是为我们这种需求量身定做。

利用 concat 的必须调用onComplete后才能订阅下一个 Observable的特性,我们就可以先读取缓存数据,倘若获取到的缓存数据不是我们想要的,再调用onComplete()以执行获取网络数据的 Observable,如果缓存数据能应我们所需,则直接调用onNext(),防止过度的网络请求,浪费用户的流量。

Observable<FoodList> cache =Observable.create(new ObservableOnSubscribe<FoodList>() {@Overridepublic void subscribe(@NonNull ObservableEmitter<FoodList> e) throws Exception {Log.e(TAG, "create当前线程:" + Thread.currentThread().getName());FoodList data = CacheManager.getInstance().getFoodListData();// 在操作符 concat 中,只有调用 onComplete 之后才会执行下一个 Observableif (data != null) { // 如果缓存数据不为空,则直接读取缓存数据,而不读取网络数据isFromNet = false;Log.e(TAG, "\nsubscribe: 读取缓存数据:");runOnUiThread(new Runnable() {@Overridepublic void run() {mRxOperatorsText.append("\nsubscribe: 读取缓存数据:\n");}});e.onNext(data);} else {isFromNet = true;runOnUiThread(new Runnable() {@Overridepublic void run() {mRxOperatorsText.append("\nsubscribe: 读取网络数据:\n");}});Log.e(TAG, "\nsubscribe: 读取网络数据:");e.onComplete();}}});Observable<FoodList> network =Rx2AndroidNetworking.get("/api/food/list").addQueryParameter("rows", 10 + "").build().getObjectObservable(FoodList.class);// 两个 Observable 的泛型应当保持一致Observable.concat(cache, network).subscribeOn(Schedulers.io()).observeOn(AndroidSchedulers.mainThread()).subscribe(new Consumer<FoodList>() {@Overridepublic void accept(@NonNull FoodList tngouBeen) throws Exception {Log.e(TAG, "subscribe 成功:" + Thread.currentThread().getName());if (isFromNet) {mRxOperatorsText.append("accept : 网络获取数据设置缓存: \n");Log.e(TAG, "accept : 网络获取数据设置缓存: \n" + tngouBeen.toString());CacheManager.getInstance().setFoodListData(tngouBeen);}mRxOperatorsText.append("accept: 读取数据成功:" + tngouBeen.toString() + "\n");Log.e(TAG, "accept: 读取数据成功:" + tngouBeen.toString());}}, new Consumer<Throwable>() {@Overridepublic void accept(@NonNull Throwable throwable) throws Exception {Log.e(TAG, "subscribe 失败:" + Thread.currentThread().getName());Log.e(TAG, "accept: 读取数据失败:" + throwable.getMessage());mRxOperatorsText.append("accept: 读取数据失败:" + throwable.getMessage() + "\n");}});

有时候我们的缓存可能还会分为 memory 和 disk ,实际上都差不多,无非是多写点Observable,然后通过concat合并即可。

flatMap 多个网络请求依次依赖

想必这种情况也在实际情况中比比皆是,例如用户注册成功后需要自动登录,我们只需要先通过注册接口注册用户信息,注册成功后马上调用登录接口进行自动登录即可。

我们的flatMap恰好解决了这种应用场景,flatMap操作符可以将一个发射数据的Observable变换为多个Observables,然后将它们发射的数据合并后放到一个单独的Observable,利用这个特性,我们很轻松地达到了我们的需求。

Rx2AndroidNetworking.get("/api/food/list").addQueryParameter("rows", 1 + "").build().getObjectObservable(FoodList.class) // 发起获取食品列表的请求,并解析到FootList.subscribeOn(Schedulers.io()) // 在io线程进行网络请求.observeOn(AndroidSchedulers.mainThread()) // 在主线程处理获取食品列表的请求结果.doOnNext(new Consumer<FoodList>() {@Overridepublic void accept(@NonNull FoodList foodList) throws Exception {// 先根据获取食品列表的响应结果做一些操作Log.e(TAG, "accept: doOnNext :" + foodList.toString());mRxOperatorsText.append("accept: doOnNext :" + foodList.toString()+"\n");}}).observeOn(Schedulers.io()) // 回到 io 线程去处理获取食品详情的请求.flatMap(new Function<FoodList, ObservableSource<FoodDetail>>() {@Overridepublic ObservableSource<FoodDetail> apply(@NonNull FoodList foodList) throws Exception {if (foodList != null && foodList.getTngou() != null && foodList.getTngou().size() > 0) {return Rx2AndroidNetworking.post("/api/food/show").addBodyParameter("id", foodList.getTngou().get(0).getId() + "").build().getObjectObservable(FoodDetail.class);}return null;}}).observeOn(AndroidSchedulers.mainThread()).subscribe(new Consumer<FoodDetail>() {@Overridepublic void accept(@NonNull FoodDetail foodDetail) throws Exception {Log.e(TAG, "accept: success :" + foodDetail.toString());mRxOperatorsText.append("accept: success :" + foodDetail.toString()+"\n");}}, new Consumer<Throwable>() {@Overridepublic void accept(@NonNull Throwable throwable) throws Exception {Log.e(TAG, "accept: error :" + throwable.getMessage());mRxOperatorsText.append("accept: error :" + throwable.getMessage()+"\n");}});

zip 多个接口数据共同更新 UI

在实际应用中,我们极有可能会在一个页面显示的数据来源于多个接口,这时候我们的zip操作符为我们排忧解难。

zip操作符可以将多个Observable的数据结合为一个数据源再发射出去。

Observable<MobileAddress> observable1 =Rx2AndroidNetworking.get("/MobilePlace/LookUp?key=ec47b85086be4dc8b5d941f5abd37a4e&mobileNumber=13021671512").build().getObjectObservable(MobileAddress.class);Observable<CategoryResult> observable2 =Network.getGankApi().getCategoryData("Android", 1, 1);Observable.zip(observable1, observable2, new BiFunction<MobileAddress, CategoryResult, String>() {@Overridepublic String apply(@NonNull MobileAddress mobileAddress, @NonNull CategoryResult categoryResult) throws Exception {return "合并后的数据为:手机归属地:" + mobileAddress.getResult().getMobilearea() + "人名:" + categoryResult.results.get(0).who;}}).subscribeOn(Schedulers.io()).observeOn(AndroidSchedulers.mainThread()).subscribe(new Consumer<String>() {@Overridepublic void accept(@NonNull String s) throws Exception {Log.e(TAG, "accept: 成功:" + s + "\n");}}, new Consumer<Throwable>() {@Overridepublic void accept(@NonNull Throwable throwable) throws Exception {Log.e(TAG, "accept: 失败:" + throwable + "\n");}});

interval 心跳间隔任务

想必即时通讯等需要轮训的任务在如今的 APP 中已是很常见,而 RxJava 2.x 的interval操作符可谓完美地解决了我们的疑惑。

这里就简单的意思一下轮训

private Disposable mDisposable;@Overrideprotected void doSomething() {mDisposable =Flowable.interval(1, TimeUnit.SECONDS).doOnNext(new Consumer<Long>() {@Overridepublic void accept(@NonNull Long aLong) throws Exception {Log.e(TAG, "accept: doOnNext : " + aLong);}}).observeOn(AndroidSchedulers.mainThread()).subscribe(new Consumer<Long>() {@Overridepublic void accept(@NonNull Long aLong) throws Exception {Log.e(TAG, "accept: 设置文本 :" + aLong);mRxOperatorsText.append("accept: 设置文本 :" + aLong + "\n");}});}/*** 销毁时停止心跳*/@Overrideprotected void onDestroy() {super.onDestroy();if (mDisposable != null) {mDisposable.dispose();}}

RxJava 1.x 如何平滑升级到 RxJava 2.x?

由于 RxJava 2.x 变化较大无法直接升级,幸运的是,官方为我们提供了 RxJava2Interrop 这个库,可以方便地把 RxJava 1.x 升级到 RxJava 2.x,或者将 RxJava 2.x 转回到 RxJava 1.x。

与RxJava 1.x的差异

给Android开发者的 RxJava 详解:该文详细地为大家讲解了 RxJava 的优势、原理以及使用方式和适用情景,一定被众多的 Android 开发者视为神器。可惜,文章历史比较久远,基本都是讲解的 RxJava 1.x了。

因为 RxJava 2.x 是按照Reactive-Streams specification规范完全的重写的,完全独立于 RxJava 1.x 而存在,它改变了以往 RxJava 的用法。

RxJava2Examples (正在更新)。

Nulls

这是一个很大的变化,熟悉 RxJava 1.x 的童鞋一定都知道,1.x 是允许我们在发射事件的时候传入 null值的,但现在我们的2.x 不支持了。

传入null会产生NullPointerException。这意味着Observable<Void>不再发射任何值,而是正常结束或者抛出空指针。

Flowable

在 RxJava 1.x 中关于介绍backpressure部分有一个小小的遗憾,那就是没有用一个单独的类,而是使用Observable。而在 2.x 中Observable不支持背压了,将用一个全新的Flowable支持背压

背压:大概就是指在异步场景中,被观察者发送事件的速度远快于观察者的处理速度的情况下,一种告诉上游的被观察者降低发送速度的策略。在差距太大的时候,我们的内存会猛增,直到OOM。而我们的 Flowable 一定意义上可以解决这样的问题,但其实并不能完全解决,这个后面可能会提到。

Single/Completable/Maybe

其实这三者都差不多,Single顾名思义,只能发送一个事件,和Observable接受可变参数完全不同。而Completable侧重于观察结果,而Maybe是上面两种的结合体。也就是说,当你只想要某个事件的结果(true or false)的时候,你可以使用这种观察者模式。

线程调度相关

这一块基本没什么改动,但细心的小伙伴一定会发现,RxJava 2.x 中已经没有了Schedulers.immediate()这个线程环境,还有Schedulers.test()

Function相关

熟悉 1.x 的小伙伴一定都知道,我们在1.x 中是有Func1Func2…..FuncN的,但 2.x 中将它们移除,而采用Function替换了Func1,采用BiFunction替换了Func 2..N。并且,它们都增加了 throws Exception,也就是说,妈妈再也不用担心我们做某些操作还需要 try-catch 了。

其他操作符相关

Func1...N的变化,现在同样用ConsumerBiConsumerAction1Action2进行了替换。后面的Action都被替换了,只保留了ActionN

官方对比

下面从官方截图展示 2.x 相对 1.x 的改动细节,仅供参考。

引用:

文章链接:

这可能是最好的RxJava 2.x 教程(完结版)

这可能是最好的RxJava 2.x 入门教程(一)

这可能是最好的RxJava 2.x 入门教程(二)

这可能是最好的RxJava 2.x 入门教程(三)

这可能是最好的RxJava 2.x 入门教程(四)

这可能是最好的RxJava 2.x 入门教程(五)

GitHub 同步代码:/nanchen2251/RxJava2Examples,主要包含基本的代码封装,RxJava 2.x所有操作符应用场景介绍和实际应用场景,后期除了RxJava可能还会增添其他东西。

本内容不代表本网观点和政治立场,如有侵犯你的权益请联系我们处理。
网友评论
网友评论仅供其表达个人看法,并不表明网站立场。
相关阅读
RxJava 2.x 入门

RxJava 2.x 入门

2022-07-06

RxJava 2.x 教程

RxJava 2.x 教程

2019-05-30

RxJava 3.x 使用总结

RxJava 3.x 使用总结

2022-08-23

RxJava 1.x使用简略

RxJava 1.x使用简略

2020-07-11