RxJava学习

什么是RxJava?

要知道什么是RxJava首先要去了解下Rx。Rx全称Reactive Extensions,直译过来就是响应式扩展。
RxJava是基于观察者模式,他是一种编程模型,目标是提供一致的编程接口,帮助开发者方便的处理异步数据流。

ReactiveX.io给的定义是,Rx是一个使用可观察数据流进行异步编程的编程接口,ReactiveX结合了观察者模式、迭代器模式和函数式编程的精华。Rx已经渗透到了各个语言中,有了Rx所以才有了 RxJava,Rx.NET、RxJS、RxSwift、Rx.rb、RxPHP等等,更详细的可以去这里看看languages

那么RxJava到底是什么呢?我对他的理解就是对于java语言的异步的响应式编程库。

简单说下什么是响应式编程

举个简单的栗子;

    A=B+C

A被赋值为B和C的和,这是如果B的值发生变化,A的值并不会发生变化。而如果我们运用一种机制,当B或者C发送变化,A的值也随之变化,这就是响应式。

关于响应式的更多更多了解点击这里传送门

聊聊RxJava的观察者模式

刚刚已经了解了RxJava是有RX衍生过来的,所以RxJava也是基于观察者模式。就好比一个害羞的小男孩,暗恋一个小女孩,女生的一举一动他都认真仔细的观察,女孩饿了,渴了。。。最后终于看着女孩被别人牵走啦。所以不能只说观察,也要行动啊。

上例中的女孩就是被观察者,男孩则是观察者。对应RxJava中,有这几个基本概念:

  • Observable (可观察者,即被观察者)
  • Observer (观察者)
  • subscribe (订阅)
  • 事件

Observable 和 Observer 通过 subscribe() 方法实现订阅关系,从而 Observable 可以在需要的时候发出事件来通知 Observer。

与传统观察者模式不同, RxJava 的事件回调方法除了普通事件 onNext() 
(相当于 onClick() / onEvent())之外,还定义了两个特殊的事件:onCompleted() 和 onError()。

onCompleted(): 事件队列完结。RxJava 不仅把每个事件单独处理,还会把它们看做一个队列。
RxJava 规定,当不会再有新的 onNext() 发出时,需要触发 onCompleted() 方法作为标志。

onError(): 事件队列异常。在事件处理过程中出异常时,onError() 会被触发,同时队列自动终止,不允许再有事件发出。

在一个正确运行的事件序列中, onCompleted() 和 onError() 有且只有一个,并且是事件序列中的最后一个。需要注意的是,onCompleted() 和 onError() 二者也是互斥的,即在队列中调用了其中一个,就不应该再调用另一个。

RxJava API的使用

(1) 创建 Observer

RxJava学习_第1张图片
observer_create.png

(2) 创建 Observable

RxJava学习_第2张图片
observable_create.png

也可以利用just或者from快速创建observable

  • just(T...): 将传入的参数依次发送出来。
    Observable observable = Observable.just("Hello", "Hi", "Aloha");
    // 将会依次调用:
    // onNext("Hello");
    // onNext("Hi");
    // onNext("Aloha");
    // onCompleted();
  • from(T[]) / from(Iterable) : 将传入的数组或 Iterable 拆分成具体对象后,依次发送出来。
    String[] words = {"Hello", "Hi", "Aloha"};
    Observable observable = Observable.from(words);
    // 将会依次调用:
    // onNext("Hello");
    // onNext("Hi");
    // onNext("Aloha");
    // onCompleted();

(3) Subscribe (订阅)

创建了 Observable 和 Observer 之后,再用 subscribe() 方法将它们联结起来,整条链子就可以工作了。代码形式很简单:

observable.subscribe(observer);
// 或者:
observable.subscribe(subscriber);

(4)线程控制Scheduler

RxJava学习_第3张图片
Schedulers.png

在调用Scheduler的时候有这两个方法

  • subscribeOn() : 设置的是被观察者所做的一些操作的线程(发送事件的线程)
  • observeOn() : 设置的是观察者被通知之后在哪个线程操作(接收事件的线程)

多次指定发送事件(上游)的线程只有第一次指定的有效, 也就是说多次调用subscribeOn() 只有第一次的有效, 其余的会被忽略.

多次指定接收(下游)的线程是可以的, 也就是说每调用一次observeOn() , 下游的线程就会切换一次.

RxJava 操作符

  • map
    Map操作符对原始Observable发射的每一项数据应用一个你选择的函数,然后返回一个发射这些结果的Observable。
RxJava学习_第4张图片
map.png

例:得到多个Student对象中的name,保存到nameList中

    Observable.just(student1, student2, student2)
                    //使用map进行转换,参数1:转换前的类型,参数2:转换后的类型
                    .map(new Func1() {
                        @Override
                        public String call(Student i) {
                            String name = i.getName();//获取Student对象中的name
                            return name;//返回name
                        }
                    })
                    .subscribe(new Action1() {
                        @Override
                        public void call(String s) {
                            nameList.add(s);
                        }
                    });

上例,map将原来的student数据源转化为String name 发送出去了,观察者拿到的就是name

  • FlatMap
    FlatMap将一个发射数据的Observable变换为多个Observables,然后将它们发射的数据合并后放进一个单独的Observable

  • concatMap
    它类似于最简单版本的flatMap,但是它按次序连接而不是合并那些生成的Observables,然后产生自己的数据序列。

RxJava学习_第5张图片
flatMap.png

例如,上面是打印每个学生的名字,这边我们需要打印出每个学生的课程名称,一个学生有多个课程

      /**
         * 学生类
         */
        class Student {
            private String name;//姓名
            private List coursesList;//所修的课程
            ...
        }
        /**
         * 课程类
         */
        class  Course {
            private String name;//课程名
            private String id;
            ...
        }

利用flatMap变化一下:

     List students = new ArrayList();
            students.add...
            ...

            Observable.from(students)
                    .flatMap(new Func1>() {
                        @Override
                        public Observable call(Student student) {
                            return Observable.from(student.getCoursesList());
                        }
                    })
                    .subscribe(new Action1() {
                        @Override
                        public void call(Course course) {
                            Log.i(TAG, course.getName());
                        }
                    });

上述例子中,数据源转化是这样的:

 student  ->  course1,course2...  -> 利用from()将所有的course放入一个observerble中发送出去

 接收消息的数据源就变成 List课程的列表,直接打印名称
  • zip
    通过一个函数将多个Observables的发射物结合到一起,基于这个函数的结果为每个结合体发射单个数据项。
RxJava学习_第6张图片
zip.png

使用场景:
比如一个界面需要展示用户的一些信息, 而这些信息分别要从两个服务器接口中获取, 而只有当两个都获取到了之后才能进行展示, 这个时候就可以用Zip了:

Compose操作符是将源Observable按照自定义的方式转化成另外一个新的Observable。可以这么说compose是对Observable进行操作的而lift是对Subscriber进行操作的,作用点是不同的。
(这里的lift类似map,将数据类型转化)

  • fliter
    只发射通过了谓词测试的数据项
RxJava学习_第7张图片
filter.png

你可能感兴趣的:(RxJava学习)