RxJava 1.x 理解-2

给RxJava 加入线程控制 -- Scheduler

在 RxJava 1.x 理解-1 中,我们说到了RxJava的简单用法,但是这还远远不够,因为这简单用法是在同一个线程中使用的。比如我们需要在子线程中进行耗时操作,在Android 主线程中进行UI操作,这样才不会导致卡顿的发生。

先上代码:

        Observable
                .create(new Observable.OnSubscribe<String>() {
                    @Override
                    public void call(Subscriber<? super String> subscriber) {
                        subscriber.onNext("提供数据01 获取线程id:" + Thread.currentThread().getId());
                        subscriber.onNext("提供数据02 获取线程id:" + Thread.currentThread().getId());
                        subscriber.onNext("提供数据03 获取线程id:" + Thread.currentThread().getId());
                        subscriber.onNext("提供数据04 获取线程id:" + Thread.currentThread().getId());
                    }
                })
                .subscribeOn(Schedulers.io())              // 订阅发生在   // 指定 subscribe() 发生在 IO 线程   
                .observeOn(AndroidSchedulers.mainThread()) // 观察发生在   // 指定 Subscriber 的回调发生在主线程 
                .subscribe(new Action1<String>() {
                    @Override
                    public void call(String s) {
                        Log.d(TAG, "Item: " + s + " 执行调用 获取线程id:" + Thread.currentThread().getId());
                    }
                });

输入的结果:

02-08 22:19:04.212 6071-6071/pers.bolin.rxjavademo D/MainActivity: Item: 提供数据01 获取线程id:6091 / 执行调用 获取线程id:2
02-08 22:19:04.212 6071-6071/pers.bolin.rxjavademo D/MainActivity: Item: 提供数据02 获取线程id:6091 / 执行调用 获取线程id:2
02-08 22:19:04.212 6071-6071/pers.bolin.rxjavademo D/MainActivity: Item: 提供数据03 获取线程id:6091 / 执行调用 获取线程id:2
02-08 22:19:04.212 6071-6071/pers.bolin.rxjavademo D/MainActivity: Item: 提供数据04 获取线程id:6091 / 执行调用 获取线程id:2

可以看出提供数据的事件发生在IO线程中,而回调的事件发生在主线程中了。

在RxJava 中,Scheduler ——调度器,相当于线程控制器,RxJava 通过它来指定每一段代码应该运行在什么样的线程。RxJava 已经内置了几个 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 主线程运行。

有了这几个 Scheduler ,就可以使用 subscribeOn() 和 observeOn() 两个方法来对线程进行控制了。 * subscribeOn(): 指定 subscribe() 所发生的线程,即 Observable.OnSubscribe 被激活时所处的线程。或者叫做事件产生的线程。 * observeOn(): 指定 Subscriber 所运行在的线程。或者叫做事件消费的线程。

 

参考资料:

一起来造一个RxJava,揭秘RxJava的实现原理

给 Android 开发者的 RxJava 详解

 

posted @ 2018-02-08 23:01  H_bolin  阅读(232)  评论(0编辑  收藏  举报