RxJava基本使用

RxJava基本使用_第1张图片
image

参考: 给 Android 开发者的 RxJava 详解
GitHub: RxJava、 RxAndroid

// Build.gradle配置文件引入
compile 'io.reactivex:rxandroid:1.1.0'
compile 'io.reactivex:rxjava:1.1.0'

1、RxJava可以做什么

RxJava是一个在 Java VM 上使用可观测的序列来组成异步的、基于事件的程序的库。可以轻松实现线程切换的方式。平时我们做线程切换的几种方式:

  • Handler
    这个应该是Android常用的一种方式,利用线程的Handler发送消息到相应的线程Looper里面,那么这个任务也将会在该线程执行。
handler.sendEmptyMessage(EVENT_DOWNLOAD_DATA);

Handler另外的一种发送消息的方式:

getUiHandler().post(new Runnable() {
    @Override
    public void run() {
        // do something                
    }
});
  • new Thread()
    新建一个子线程执行耗时操作,需要切换到主线更新UI时
getActivity().runOnUiThread(new Runnable() {
    @Override
    public void run() {
         // do something       
    }
});
  • AsyncTask
    内部有个线程池执行异常任务doInBackground(),在UI线程中执行任务前onPreExecute()、更新进度onProgressUpdate()、完成任务onPostExecute()的方法。
  • EventBus、otto
    比较通用的事件消息总线开源框架,对于普通的线程切换来说,也是比较重的一种方式。
  • 另外还有广播、Service等不大灵活的线程切换方式。

相必,以上的几种方式使用起来都不是那么的简洁,业务复杂起来,还会导致逻辑上的不清晰,那么RxJava就可以做到逻辑的简洁,执行异步任务、线程切换都可以在一条链上完成。

2、观察者模式

观察者模式的例子:

  • Button点击事件
    Button被观察者,持有引用,OnClickListener观察者,通过setOnClickListener()注册/订阅点击事件,发生onClick()事件。
  • 设置监听回调函数
    比如执行选择会员弹窗Dialog。Dialog被观察者,持有引用,OnMemberSelectListener观察者,通过setOnMemberSelectListener()注册/订阅选择会员事件,发生onMemberSelect(Member member)事件。

3、RxJava观察者模式实现步骤

3.1、创建观察者 Observer

Observable被观察者 和 Observer观察者 通过 subscribe() 方法实现订阅关系
事件。

onNext()
onCompleted()
onError()
onStart() // 这是 Subscriber 增加的方法。它总是在 subscribe 所发生的线程被调用。不太适用于执行任务前的弹窗等UI操作。
unsubscribe() // 这是 Subscriber 所实现的另一个接口 Subscription 的方法,用于取消订阅。避免内存泄漏。

Observer 接口,Observer 的抽象类:Subscriber

Subscriber subscriber = new Subscriber() {
    @Override
    public void onNext(String s) {
        Log.d(tag, "Item: " + s);
    }

    @Override
    public void onCompleted() {
        Log.d(tag, "Completed!");
    }

    @Override
    public void onError(Throwable e) {
        Log.d(tag, "Error!");
    }
};
3.2、创建被观察者 Observable

create() 方法是 RxJava 最基本的创造事件序列的方法

Observable observable = Observable.create(new Observable.OnSubscribe() {
    @Override
    public void call(Subscriber subscriber) {
        subscriber.onNext("Hello");
        subscriber.onNext("Hi");
        subscriber.onNext("Aloha");
        subscriber.onCompleted();
    }
});

另外还提供了一些方法用来快捷创建事件队列

// just()
Observable observable = Observable.just("Hello", "Hi", "Aloha");

// from()
String[] words = {"Hello", "Hi", "Aloha"};
Observable observable = Observable.from(words);
3.3、Subscribe (订阅)
observable.subscribe(observer);
// 或者:
observable.subscribe(subscriber);

观察者被观察者换位置:对流式 API 的设计会造成影响。

在 RxJava 中, Observable 并不是在创建的时候就立即开始发送事件,而是在它被订阅的时候。

除了 subscribe(Observer) 和 subscribe(Subscriber) ,subscribe() 还支持不完整定义的回调

// 自动创建 Subscriber ,并使用 onNextAction 来定义 onNext()
observable.subscribe(onNextAction);
// 自动创建 Subscriber ,并使用 onNextAction 和 onErrorAction 来定义 onNext() 和 onError()
observable.subscribe(onNextAction, onErrorAction);
// 自动创建 Subscriber ,并使用 onNextAction、 onErrorAction 和 onCompletedAction 来定义 onNext()、 onError() 和 onCompleted()
observable.subscribe(onNextAction, onErrorAction, onCompletedAction);

Action0 是 RxJava 的一个接口,它只有一个方法 call(),这个方法是无参无返回值的;由于 onCompleted() 方法也是无参无返回值的,因此 Action0 可以被当成一个包装对象,将 onCompleted() 的内容打包起来将自己作为一个参数传入 subscribe() 以实现不完整定义的回调。这样其实也可以看做将 onCompleted() 方法作为参数传进了 subscribe(),相当于其他某些语言中的『闭包』。

4、RxJava操作符

变换:就是将事件序列中的对象或整个序列进行加工处理,转换成不同的事件或事件序列

FuncX 和 ActionX 的区别在 FuncX 包装的是有返回值的方法。

4.1、map()

事件对象的直接变换

Observable.from(students)
    .map(new Func1() {
        @Override
        public String call(Student student) {
            return student.getName();
        }
    })
    .subscribe(subscriber);
4.2、flatMap()
Observable.from(students)
    .flatMap(new Func1>() {
        @Override
        public Observable call(Student student) {
            return Observable.from(student.getCourses());
        }
    })
    .subscribe(subscriber);

flatMap() 中返回的是个 Observable 对象,并且这个 Observable 对象并不是被直接发送到了 Subscriber 的回调方法中。 flatMap() 的原理是这样的:

    1. 使用传入的事件对象创建一个 Observable 对象;
    1. 并不发送这个 Observable, 而是将它激活,于是它开始发送事件;
    1. 每一个创建出来的 Observable 发送的事件,都被汇入同一个 Observable ,而这个 Observable 负责将这些事件统一交给 Subscriber 的回调方法。

由于可以在嵌套的 Observable 中添加异步代码, flatMap() 也常用于嵌套的异步操作,例如嵌套的网络请求。

5、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。
  • AndroidSchedulers.mainThread(),它指定的操作将在 Android 主线程运行。
  • subscribeOn():subscribe() 所发生的线程,即Observable.OnSubscribe 被激活时所处的线程。或者叫做事件产生的线程。
  • observeOn():Subscriber 所运行在的线程。或者叫做事件消费的线程。
Observable.just(1, 2, 3, 4) // IO 线程,由 subscribeOn() 指定
    .subscribeOn(Schedulers.io())
    .observeOn(Schedulers.newThread())
    .map(mapOperator) // 新线程,由 observeOn() 指定
    .observeOn(Schedulers.io())
    .map(mapOperator2) // IO 线程,由 observeOn() 指定
    .observeOn(AndroidSchedulers.mainThread) 
    .subscribe(subscriber);  // Android 主线程,由 observeOn() 指定

通过 observeOn() 的多次调用,程序实现了线程的多次切换。

不过,不同于 observeOn() , subscribeOn() 的位置放在哪里都可以,但它是只能调用一次的。

  • doOnSubscribe()

与 Subscriber.onStart() 相对应的,有一个方法 Observable.doOnSubscribe() 。它和 Subscriber.onStart() 同样是在 subscribe() 调用后而且在事件发送前执行,但区别在于它可以指定线程。默认情况下, doOnSubscribe() 执行在 subscribe() 发生的线程;而如果在 doOnSubscribe() 之后有 subscribeOn() 的话,它将执行在离它最近的 subscribeOn() 所指定的线程。

Observable.create(onSubscribe)
    .subscribeOn(Schedulers.io())
    .doOnSubscribe(new Action0() {
        @Override
        public void call() {
            progressBar.setVisibility(View.VISIBLE); // 需要在主线程执行
        }
    })
    .subscribeOn(AndroidSchedulers.mainThread()) // 指定主线程
    .observeOn(AndroidSchedulers.mainThread())
    .subscribe(subscriber);

你可能感兴趣的:(RxJava基本使用)