1.Scheduler 的 API
在响应式编程中,如果想给Observable操作符链添加多线程功能,可以指定操作符(或者特定的Observable)在特定的调度器(Scheduler)上执行。
RxJava提供了5种调度器:
- Schedulers.immediate():这个调度器允许立即在当前线程执行所指定的工作。它是默认的调度器。
- Schedulers.newThread():总是启用新线程,并在新线程执行操作。
- Schedulers.io():这个调度器适用于I/O操作。它基于根据需要,增长或缩减来自适应的线程池。因此多数情况下 io() 比 newThread() 更有效率。不要把计算工作放在 io() 中,可以避免创建不必要的线程。
- Schedulers.computation():这个是计算工作默认的调度器,这个计算指的是 CPU 密集型计算,即不会被 I/O 等操作限制性能的操作。这个 Scheduler 使用的固定的线程池,大小为 CPU 核数。不要把 I/O 操作放在 computation() 中,否则 I/O 操作的等待时间会浪费 CPU。
- Schedulers.trampoline():当我们想在当前线程执行一个任务时,并不是立即,我们可以用.trampoline()将它入队。这个调度器将会处理它的队列并且按序运行队列中每一个任务。
补充:Android 还有一个专用的 AndroidSchedulers.mainThread(),它指定的操作将在 Android 主线程运行。
那么如何进行线程调度呢?
可以使用ObserveOn()和subscribeOn()操作符,让Observable在一个特定的调度器上执行。
- subscribeOn(): 指定 subscribe()所发生的线程,即 Observable.OnSubscribe 被激活时所处的线程。或者叫做事件产生的线程。
- observeOn(): 指定 Subscriber 所运行在的线程。或者叫做事件消费的线程。
2.举个栗子(代码演示)
//模拟网络下载耗时操作
private List<String> getData(){
List<String> array = new ArrayList<String>();
//加载10个数据
for (int i=0;i<10;i++){
try {
Thread.sleep(500);
} catch (InterruptedException e) {
e.printStackTrace();
}
array.add(i+"");
}
return array;
}
private void testRxJava(){
rx.Observable.from(getData())
.subscribeOn(Schedulers.io())
.observeOn(AndroidSchedulers.mainThread())
.subscribe(new Action1<String>() {
@Override
public void call(String s) {
Log.d(TAG,">>>item:"+s);
}
});
}
补充:Scheduler切换线程中,通过 observeOn() 的多次调用,程序可以在线程间多次切换。而subscribeOn() 只能调用一次的,它可以放在任何位置。
3.Scheduler 的原理
RxJava 遵循的是线程不变的原则,即:在哪个线程调用 subscribe(),就在哪个线程生产事件;在哪个线程生产事件,就在哪个线程消费事件。如果需要切换线程,就需要用到 Scheduler (调度器)。
- subscribeOn() 原理图:
subscribeOn()通过操作特定的Scheduler,通知Observable在指定线程执行。
- observeOn() 原理图:
ObserveOn()通过操作特定的Scheduler,让Observable指定Observers的执行线程;
SubscribeOn():指定了事件源将要执行的线程,无论在操作符链的那个位置被调用;因此,不管SubscribeOn()方法调用了多少次,只有离事件源最近的一个会生效,其他都被最近的一个截断了。
ObserveOn():指定的是方法调用以后,Observers所执行的线程。因此,我们可以在操作符链中,多次调用ObserveOn()来切换执行的线程。