RXjava 中的observeOn subscribeOn 两个函数十分类似,相信很多人在使用时会有疑问。官方文档的说明,包括图片不是那么容易理解,特此立翻译贴一篇。
官方文档解释如下地址:
http://reactivex.io/documentation/operators/observeon.html
http://reactivex.io/documentation/operators/subscribeon.html
================================= observeOn 篇 ========================================
指定观察者运行的Scheduler(Scheduler即指定的所在线程)
(图片解释:重点是箭头,初始化黑色箭头代表默认线程,调用observeOn方法 黄色箭头指定Scheduler,所以下面的箭头都变成了黄色,即改变了代码运行所在的线程)
很多基于ReactiveX 的实现使用Schedulers来控制一个被观察者在多个线程之间切换。你可以通过observeOn操作来指定一个Scheduler,命令一个被观察者通过此Scheduler发送通知给观察者。
(图解:这个图很牛逼!先表扬一下......我们从上往下看,初始化(xian)箭(cheng)头颜色不再跟第一张图是黑色的了,而是蓝色的,我们发现下面有个subscribeOn 方法是指定蓝色(xian)箭(cheng)头的,没错,原因就在这里,具体为什么等会翻译subscribeOn的时候再说。然后是一个observeOn方法指定黄色(xian)箭(cheng)头,然后是一个map操作,(xian)箭(cheng)头是黄色的,然后是subscribeOn操作,暂时略过,最后又是observeOn操作,指定的是粉色(xian)箭(cheng)头,这个操作之后,黄色(xian)箭(cheng)头变为了粉色(xian)箭(cheng)头。)
subscribeOn操作是类似的,但是subscribeOn是控制被观察者在哪一个Scheduler上执行自身的操作(onsubscribe的call方法)以及通知观察者的操作(onNext,onCompleted,map等)。
默认情况下,被观察者以及被观察者的调用链操作会在调用subscribe方法的线程上执行。subscribeOn操作能指定一个特殊的Scheduler,从而改变这个行为,使被观察者的操作在这个Scheduler上执行。observeOn操作指定一个不同的Scheduler,被观察者会在这个Scheduler上发送通知给观察者。
在上述图中,subscribeOn操作指定被观察者在哪一个线程执行操作,不论在调用链中哪一个位置调用这个方法。
observeOn影响observeOn操作之后的被观察者运行的线程。因为这个原因,你可以在调用链中多次调用observeOn来达到改变运行的所在线程。
================================= subscribeOn篇 ========================================
指定被观察者运行的Scheduler
(图解:区别上一篇,在调用subscribeOn方法后,初始化的线程也是黄色箭头的,所以两者的区别在于作用域不同,subscribeOn影响整个被观察者的运行所在线程,observeOn影响的是observeOn操作后的代码)
很多基于ReactiveX 的实现使用Schedulers来控制一个被观察者在多个线程之间切换。你可以通过subscribeOn操作来指定一个Scheduler,命令一个被观察者通过此Scheduler执行工作。(对比上文observeOn中此处的解释,理解他们之间的区别)
observeOn操作也是相似的,但是observeOn操作局限性大一点。observeOn仅仅影响被观察者在哪个Schedulers上发送通知给观察者。
(接下去的内容跟observeOn篇是一样的)
========================================== 补充 =========================================
看完文章,我们明白observeOn可以多次调用,调度不同线程执行操作,但对于subscribeOn多次调用的情况下会如何,官方文档中并未提及。
从源码中分析,subscribeOn会生成一个新的被观察者对象,老的被观察者会被新对象持有,类似包装类模式,新对象对老对象的操作就是将老对象丢到指定的新线程中去执行,所以!不管你调用几次subscribeOn,最终还是会回到第一次调用subscribeOn指定的线程中去。(包装类不管包装几次,一次一次往里调用,最终还是会调用到最根本的那一个类中去)