RxJava 2.x 之学后感觉会了但还是一脸懵逼总结

前言

应该有小伙伴和我一样吧,在学习了南尘大大的'这可能是最好的RxJava 2.x 入门教程系列专栏'后,虽然对RxJava的使用都会有一定程度认识了,但是南尘大大的完结篇怎么打不开啊o(╥﹏╥)o,而且南尘大大在介绍操作符的时候都是简单介绍加代码实现,对此我还是存在一些疑惑和不理解的,加上如果不是常用的话,这么多操作符还是很容易忘记的,所以就来写一篇总结吧!

这里给出南尘大神的文章链接:
这可能是最好的 RxJava 2.x 入门教程(一)
这可能是最好的 RxJava 2.x 入门教程(二)
这可能是最好的 RxJava 2.x 入门教程(三)
这可能是最好的 RxJava 2.x 入门教程(四)
这可能是最好的 RxJava 2.x 入门教程(五)
抛物线大神的文章链接:
给 Android 开发者的 RxJava 详解

正文

RxJava 到底是什么

GitHub 主页上的介绍是这样的"a library for composing asynchronous and event-based programs using observable sequences for the Java VM"(一个在 Java VM 上使用可观测的序列来组成异步的、基于事件的程序的库)。这就是 RxJava ,概括得非常精准。
通俗点说就是一个可以帮你解决异步迷之缩进烦恼的一个程序库,它和你之前学习的基本操作符是一样的,只是它比较高级一点。

RxJava 好在哪

一个词:简洁
一段话:在我们编程的时候保持我们的逻辑思路的简洁,在维护时帮助我们更容易理解。

说了跟没说一样之一脸嫌弃.png

好了好了,不要在用鄙视的眼神看我(*❦ω❦) ,我说了这是总结!总结!!下面我们直接上代码。

Observable.just(1, 2, 3).subscribeOn(Schedulers.io()).observeOn(AndroidSchedulers.mainThread()).subscribe(new Consumer<Integer>() {
      @Override
      public void accept(Integer integer) throws Exception {
          /* 业务处理... */

      }
});

这是RxJava最常见的使用或者说是结构,分为被观察者(发射器)操作符(N个)发射调度方接收调度方观察者(接收器)

Observable(发射器)相关

  • Obesevable 没什么好说的,最常用于创建被观察者的类。
  • Flowable 该类与Obeservable的区别是它支持背压,使用方式基本相同。背压是指在异步场景中,被观察者发射事件的速度远快于观察者的处理速度的情况下,一种告诉上游的被观察者降低发射速度的策略。
  • Single/Completable/Maybe 其实这三者都差不多,Single顾名思义,只能发射一个事件,和 Observable接受可变参数完全不同。而 Completable 侧重于观察结果,而 Maybe 是上面两种的结合体。也就是说,当你只想要某个事件的结果(true or false)的时候,你可以使用这种观察者模式。

Observer(接收器)相关

  • Observer 处理onSubscribe()onNext()onError()onComplete()四个方法的回调实现。
    • onSubscribe():返回接收器的控制器(Disposable),控制事件的接收。
    • onNext():接收事件并处理。
    • onCompleted(): 事件队列完结。RxJava 不仅把每个事件单独处理,还会把它们看做一个队列。RxJava 规定,当不会再有新的 onNext() 发出时,需要触发 onCompleted() 方法作为标志。
    • onError(): 事件队列异常。在事件处理过程中出异常时,onError()会被触发,同时队列自动终止,不允许再有事件发出。
    • 在一个正确运行的事件序列中, onCompleted()onError() 有且只有一个,并且是事件序列中的最后一个。需要注意的是,onCompleted()onError() 二者也是互斥的,即在队列中调用了其中一个,就不应该再调用另一个。

线程调度相关

线程控制——Scheduler(调度器):在不指定线程的情况下, RxJava 遵循的是线程不变的原则,即:在哪个线程调用 subscribe(),就在哪个线程生产事件;在哪个线程生产事件,就在哪个线程消费事件。如果需要切换线程,就需要用到 Scheduler (调度器)。

  • Schedulers.immediate(): 直接在当前线程运行,相当于不指定线程。这是默认的 Scheduler
  • Schedulers.newThread(): 总是启用新线程,并在新线程执行操作。
  • Schedulers.io(): I/O 操作(读写文件、读写数据库、网络信息交互等)所使用的 Scheduler。行为模式和 newThread() 差不多,区别在于 io() 的内部实现是是用一个无数量上限的线程池,可以重用空闲的线程,因此多数情况下 io()newThread() 更有效率。不要把计算工作放在 io() 中,可以避免创建不必要的线程。
  • Schedulers.computation(): 计算所使用的Scheduler。这个计算指的是 CPU 密集型计算,即不会被 I/O 等操作限制性能的操作,例如图形的计算。这个 Scheduler 使用的固定的线程池,大小为 CPU 核数。不要把 I/O 操作放在 computation() 中,否则 I/O 操作的等待时间会浪费 CPU。
  • 另外, Android 还有一个专用的 AndroidSchedulers.mainThread(),它指定的操作将在 Android 主线程运行。

操作符相关

创建Observables

创建一个新的源Observable

  • Create — 操作符应该是最常见的操作符了,主要用于产生一个Obserable被观察者对象。
  • Defer — 每次订阅都会创建一个新的Observable,并且如果没有被订阅,就不会产生新的Observable
  • Empty/Never/ThrowEmpty是创建一个不发射且可终止itemObservableNever是创建一个不发射且不终止itemObservableThrow是创建一个不发射且已错误终止的itemObservable
  • From — 从数组或者序列化集合中获取数据创建Observable
  • Interval — 创建一个Observable用于间隔时间执行某个操作,其接受三个参数,分别是第一次发射延迟,间隔时间,时间单位,默认由新线程调度发射。
  • Just — 从对象数组中获取并创建Observable
  • Range — 创建一个发射特定范围的连续整数的itemObservable
  • Repeat — 创建一个可以多次发射特定itemObservable
  • Timer — 创建一个延时的Observable,并只发出一次item,默认由新线程调度发射。

变换Observables

变换从源Observable已发出的item

  • Buffer — 将源Observable中的数据按skip(步长) 分成最大不超过count的迭代器集合,然后生成一个Observable,发出的迭代器集合个数 = items的个数 - skip (步长)。
  • FlatMap — 把一个源Observable通过某种方法转换为多个Observable,然后再把这些分散的Observable结合到同一个Observable,无法保证发射顺序。需要顺序发射可使用concatMap操作符。
  • GroupBy — 对源Observableitem进行分组,得到一个分组的Observable(结果集)进行发射,可通过getKey()方法获取结果集的Key值。注意,但你不对结果集处理时(既不订阅执行也不对其进行别的操作符运算),应使用take(0)操作符,以防止内存泄漏。
  • Map — 将已发射的item根据函数变化。
  • Scan — 通过函数方法进行两两递归处理每一个源Observableitem,得到最后的item发射。该操作符与reduce作用类似,但是它不仅关注结果,也关注过程。
  • Window — Window与Buffer相似 ,但不是从源Observable发射items,而是发射Observable,每个Observable都是从源Observable发出一个子item,直到收到onCompleted 通知终止。

过滤Observables

从源Observable选择性的发射item

  • Debounce — 去除发射频率过快的item
  • Distinct — 去掉重复的item
  • ElementAt — 仅发射指定索引的item
  • Filter — 常用的过滤操作符,通过test()函数过滤。
  • First — 只发射第一个item或符合条件的第一个item
  • IgnoreElements — 忽略所有源Observable发射的item,只接收onCompletedonError事件。
  • Last — 只发射最后一个item或者符合条件的最后一个item
  • Sample — 在指定的时间间隔范围内对源Observable发射item的最后一个进行周期性采样。
  • Skip — 跳过count个数目的item
  • SkipLast — 跳过倒序count个数目的item
  • Take — 最多接收count个数目的item
  • TakeLast — 最多接收倒序count个数目的item

结合Observables

结合多个源Observable去创建一个Observable

  • And/Then/When — 作用和zip()类似,使用PatternPlan作为中介,将发射的数据集合并到一起,来完成一些复杂的结构。
  • CombineLatest — 将两个源Observable中的其中的一个发射的item与另一个源Observable中最后发射的item以一定的规则进行合并。
  • Join — 该操作符与CombineLatest的作用类似,不同之处在于它可以控制每个Observable发射的生命周期。
  • Merge — 结合多个源Observable,不需要等待其中发射器的发射完毕就可以发射其他的发射器的item,可接受可变参数,也支持迭代器集合。
  • StartWith — 在源Observable发射item前,插入指定的一些数据。
  • Switch — 能够从一个Observable自动取消订阅来订阅一个新的Observable(停止旧的Observable并订阅新的Observable)。
  • Zip — 合并多个源Observable,最终合并的item数目与源Obeservableitem数目最少的相同。

处理错误的操作符

处理错误通知,让Observable能够正常终止。

  • Catch — 类似于java中的try/catch,拦截onError的调用,让Observable不会因为错误的产生而终止。
  • Retry — 当源Observable遇到错误,重新订阅它并期望它能正常终止。需指定最多重新订阅的次数。注意由于重新订阅,可能会造成数据项重复。

公用操作符

一些辅助处理的工具操作符。

  • Delay — 延迟一段指定的时间再发射item,注意当接收到onError通知时,会立即终止,如需延迟订阅源Observable,需使用delaySubscription操作符。
  • Do — 在Observable的生命周期期间执行一些操作。(常用的有doOnNextdoOnCompletedoOnError
  • Materialize/Dematerializematerialize将来自源Observable的通知(onNext/onError/onComplete)都转换为一个Notification对象,然后再按原来的顺序一次发射出去。dematerialize则相反。
  • ObserveOn — 指定Observable自身在哪个调度器上执行(即在那个线程上运行)。
  • Serialize — 强制Observable按次序发射数据并且要求功能是完好的。
  • Subscribe — 订阅,使ObervableObserver联合起来。
  • SubscribeOn — 指定Observable在哪个调度器上发送通知给观察者(调用观察者的onNext,onCompleted,onError方法)。
  • TimeInterval — 拦截源Observable发射的item,转换为两次发射的时间间隔的TimeInterval对象。
  • Timeout — 如果原始Observable过了指定的一段时长没有发射任何itemTimeout操作符会以一个onError通知终止这个Observable,或者继续一个备用的Observable
  • Timestamp — 将源Observable发射的item转换为一个包含发射时间的Timestamped对象。
  • Using — 指示Observable创建一个只在它的生命周期内存在的资源,当Observable终止时这个资源会被自动释放。

条件和布尔操作符

检测Observable发射的item的操作符。

  • All — 判定源Observable发射的所有item是否都满足某个条件,不满足则抛弃。
  • Amb — 传入多个源Observable时,只会发射其中先发射item或通知(onErroronCompleted)的Observable的所有数据。
  • Contains — 判断源Observable发射的所有item中,是否包含Contains操作符的传入值,不存在的抛弃。
  • DefaultIfEmpty — 当源Observable没有发射任何item时,使用defaultIfEmpty操作符会为你发射一个你提供默认值。
  • SequenceEqual — 判定两个源Observables是否发射相同的数据序列。(相同的数据,相同的顺序,相同的终止状态)
  • SkipUntil — 忽略源Observable发射的item直到指定的Observable开始发射的那一刻,源Observable才开始正常发射。
  • SkipWhile — 忽略源Observable发射的item直到指定的某个条件变为false的那一刻,源Observable才开始正常发射。
  • TakeUntil — 源Observable正常发射的item直到指定的Observable开始发射或者发射终止通知(onError/onComplete)的那一刻,源Observable才停止发射并终止。
  • TakeWhile — 源Observable正常发射的item直到指定的某个条件变为false的那一刻,源Observable才停止发射并终止。

数学运算及聚合操作符

对整个序列进行操作的操作符。

  • Average — 计算源Observable中所有item的平均值并发射。
  • Concat — 从多个源Observable中发射,排序接收。
  • Count — 计算源Observable中所有item的个数并发射。
  • Max — 计算源Observable中所有item的最大值并发射。
  • Min — 计算源Observable中所有item的最小值并发射。
  • Reduce — 通过函数方法处理每一个源Observableitem,得到最后的item发射。该操作符与scan作用类似,但是它只关注结果。
  • Sum — 计算源Observable中所有item的和并发射。

背压操作符

  • backpressure operators — 应对源Observable发射item的速率太快,导致Observer无法及时消费而产生异常的策略。
  • onBackpressureBuffer — 若源Observable发射item的速率太快,导致Observer无法及时消费时,缓存所有当前无法消费的item,直到Observer可以处理为止。
  • onBackpressureDrop — 若源Observable发射item的速率太快,导致Observer无法及时消费时,则将当前item抛弃。

可连接的Observable操作符

可动态控制发射的Observable操作符。

  • Connect — 指示一个ConnectableObservable开始发射数据。ConnectableObservable在被订阅时并不开始发射数据,只有在它的connect()被调用时才开始。
  • Publish — 将普通的Observable转换为ConnectableObservableConnectableObservableObservable的子类。
  • RefCount — 将普通的ConnectableObservable转换为Observable
  • Replay — 缓存Observable订阅之前已经发射的数据,这样即使有Observable在其发射数据开始之后进行订阅也能收到之前发射过的数据。Replay操作符能指定缓存的大小或者时间,这样能避免耗费太多内存。

转换操作符

  • To — 将源Observable转换为另一个对象或数据结构(迭代器集合),如果原Observable发射完他的数据需要一段时间,使用To操作符得到的Observable将阻塞等待原Observable发射完后再将数据序列打包后发射出去。

好了,终于完工了,写这篇文章的初衷只是为了自己方便记忆,希望能帮助到各位小伙伴~~ 撒花~!✿✿ヽ(°▽°)ノ✿✿

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,390评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,821评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,632评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,170评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,033评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,098评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,511评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,204评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,479评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,572评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,341评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,213评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,576评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,893评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,171评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,486评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,676评论 2 335

推荐阅读更多精彩内容

  • 我从去年开始使用 RxJava ,到现在一年多了。今年加入了 Flipboard 后,看到 Flipboard 的...
    Jason_andy阅读 5,435评论 7 62
  • 转一篇文章 原地址:http://gank.io/post/560e15be2dca930e00da1083 前言...
    jack_hong阅读 897评论 0 2
  • 前言我从去年开始使用 RxJava ,到现在一年多了。今年加入了 Flipboard 后,看到 Flipboard...
    占导zqq阅读 9,156评论 6 151
  • 我在苏州的街头,遇到一只老鼠 红蔚 我在苏州的街头看到河流 有光滑的台阶伸到水里 一只老鼠,正站在台阶上观望水 恰...
    红蔚阅读 283评论 2 5
  • 今日偶得空,忽忆往昔钓鱼闲趣,遂携两小友于深山流水中垂钓。 江流有声,断岸千尺,婉转东去。昔闻人言,其水...
    若怀_阅读 510评论 2 0