Rxjava2入门教程二:Observable与Observer响应式编程在Rxjava2中的典型实现

如需下载源码,请访问
https://github.com/fengchuanfang/Rxjava2Tutorial
本篇文章对应的Demo类为:com.edward.edward.Rxjava2Demo.Rxjava2_2_Observable
文章原创,转载请注明出处:
Rxjava2入门教程二:Observable与Observer响应式编程在Rxjava2中的典型实现

运行源码,打开app,点击Demo图标,可进入以下页面

RxjavaTutorial.png


在RxJava中,函数响应式编程具体表现为一个观察者(Observer)订阅一个可观察对象(Observable),通过创建可观察对象发射数据流,经过一系列操作符(Operators)加工处理和线程调度器(Scheduler)在不同线程间的转发,最后由观察者接受并做出响应的一个过程
ObservableSource与Observer是RxJava2中最典型的一组观察者与可观察对象的组合,其他四组可以看做是这一组的改进版或者简化版。

Observable

抽象类Observable是接口ObservableSource下的一个抽象实现,我们可以通过Observable创建一个可观察对象发射数据流。

     Observable<String> observable = Observable.create(new ObservableOnSubscribe<String>() {
            @Override
            public void subscribe(ObservableEmitter<String> emitter) throws Exception {
                emitter.onNext("Hello World");
                emitter.onComplete();
            }
        });

调用Observable.create方法,创建一个可观察对象,并通过onNext发送一条数据“Hello World”,然后通过onComplete发送完成通知。

Observer

创建一个观察者Observer来接受并响应可观察对象发射的数据

Observer<String> observer = new Observer<String>() {
            @Override
            public void onSubscribe(Disposable d) {

            }

            @Override
            public void onNext(String s) {
                System.out.println(s);
            }

            @Override
            public void onError(Throwable e) {
                e.printStackTrace();
            }

            @Override
            public void onComplete() {
                System.out.println("接受完成");
            }
        };

在onNext方法中接收到可观察对象发射的数据"Hello World",并做出响应——打印到控制台。

Observer订阅Observable

observable.subscribe(observer);

通过subscribe方法,使Observer与Observable建立订阅关系,Observer与Observable便成为了一个整体,Observer便可对Observable中的行为作出响应。

Emitter/Observer

通过Observable.create创建可观察对象时,我们可以发现具体执行发射动作的是由接口ObservableEmitter的实例化对象完成的,而ObservableEmitter<T> 继承自 接口Emitter<T>,查看源码接口Emitter的具体代码如下:

public interface Emitter<T> {
        //用来发送数据,可多次调用,每调用一次发送一条数据
    void onNext(@NonNull T value);
        //用来发送异常通知,只发送一次,若多次调用只发送第一条
    void onError(@NonNull Throwable error);
        //用来发送完成通知,只发送一次,若多次调用只发送第一条
    void onComplete();
}

onNext:用来发送数据,可多次调用,每调用一次发送一条数据
onError:用来发送异常通知,只发送一次,若多次调用只发送第一条
onComplete:用来发送完成通知,只发送一次,若多次调用只发送第一条

onError与onComplete互斥,两个方法只能调用一个不能同时调用

数据在发送时,出现异常可以调用onError发送异常通知也可以不调用,因为其所在的方法subscribe会抛出异常,

若数据在全部发送完之后均正常,可以调用onComplete发送一条完成通知

接口Observer中的三个方法(onNext,onError,onComplete)正好与Emitter中的三个方法相对应,对于Emitter中对应方法发送的数据或通知进行响应。

步骤简化

去掉中间变量可以对之前的代码简化为以下形式:

    public void demo2() {
        Observable
                .create(new ObservableOnSubscribe<String>() {
                    @Override
                    public void subscribe(ObservableEmitter<String> emitter) throws Exception {
                        emitter.onNext("Hello World");
                        emitter.onComplete();
                    }
                })
                .subscribe(new Observer<String>() {
                    @Override
                    public void onSubscribe(Disposable d) {
                    }

                    @Override
                    public void onNext(String s) {
                        System.out.println(s);
                    }

                    @Override
                    public void onError(Throwable e) {
                        e.printStackTrace();
                    }

                    @Override
                    public void onComplete() {
                        System.out.println("接受完成");
                    }
                });
    }

再应用Rxjava中强大的操作符,可以将代码简化成以下形式:

    public void demo3() {
        Observable.just("Hello World")
                .subscribe(new Consumer<String>() {
                    @Override
                    public void accept(@NonNull String s) throws Exception {
                        System.out.println(s);
                    }
                });
    }

再通过λ表达式,可进一步简化

    public void demo3_1() {
        Observable.just("Hello World").subscribe(System.out::println);
    }

其中,just操作符可用来发送单条数据,数字,字符串,数组,对象,集合都可以当做单条数据发送。
Consumer可以看做是对观察者Observer功能单一化之后的产物——消费者,上例中的Consumer通过其函数accept只接收可观察对象发射的数据,不接收异常信息或完成信息。
如果想接收异常信息或完成信息可以用下面的代码:

    public void demo4() {
        Observable.just("Hello World")
                .subscribe(new Consumer<String>() {
                    @Override
                    public void accept(@NonNull String s) throws Exception {
                        System.out.println(s);
                    }
                }, new Consumer<Throwable>() {
                    @Override
                    public void accept(@NonNull Throwable throwable) throws Exception {
                        throwable.printStackTrace();
                    }
                }, new Action() {
                    @Override
                    public void run() throws Exception {
                        System.out.println("接受完成");
                    }
                });
    }

第二个参数Consumer规定泛型<Throwable>通过函数accept接收异常信息。
第三个参数Action也是对观察者Observer功能单一化之后的产物--行动,通过函数run接收完成信息,作出响应行动。

发送数据序列

Observable可以发送单条数据也可以发送数据序列
通过最基础的方法发送数据序列:

public void demo5(final List<String> list) {
        Observable.create(new ObservableOnSubscribe<String>() {
            @Override
            public void subscribe(ObservableEmitter<String> emitter) throws Exception {
                try {
                    for (String str : list) {
                        emitter.onNext(str);
                    }
                    emitter.onComplete();
                } catch (Exception e) {
                    emitter.onError(e);
                }
            }
        }).subscribe(new Observer<String>() {
            @Override
            public void onSubscribe(Disposable d) {
            }

            @Override
            public void onNext(String s) {
                System.out.println(s);
            }

            @Override
            public void onError(Throwable e) {
                e.printStackTrace();
            }

            @Override
            public void onComplete() {
                System.out.println("接受完成");
            }
        });
    }

在subscribe方法中,遍历集合list中的String元素,通过emitter.onNext(str)逐一发送;发送完成后通过emitter.onComplete()发送完成通知;如果发送过程中遇到异常,通过emitter.onError(e)发送异常信息。
Observer中通过onNext接收emitter发送的每一条信息并打印到控制台(emitter发送几次,Observer便接收几次),通过onError(Throwable e)接收异常信息,onComplete()接收完成信息。
同样可以通过操作符对其进行简化,如下;

    public void demo6(final List<String> list) {
        Observable
                .fromIterable(list)
                .subscribe(new Consumer<String>() {
                    @Override
                    public void accept(@NonNull String s) throws Exception {
                        System.out.println(s);
                    }
                });
    }

再用λ表达式,进一步简化

    public void demo6_1(final List<String> list) {
        Observable.fromIterable(list).subscribe(System.out::println);
    }

其中fromIterable操作符,可用来将一个可迭代对象中的元素逐一发送

Disposable

在之前的例子中,可以看到Observer接口中还有一个方法

public void onSubscribe(Disposable d) {
}

是在观察者Observer与可观察对象Observable,建立订阅关系后,回调这个方法,并且传过来一个Disposable类型的参数,可通过Disposable来控制Observer与Observable之间的订阅。
无论观察者Observer以何种方式订阅可观察对象Observable,都会生成一个Disposable,如下:

public void demo7(final List<String> list) {
        Disposable disposable1 = Observable.just("Hello World")
                .subscribe(new Consumer<String>() {
                    @Override
                    public void accept(@NonNull String s) throws Exception {
                        System.out.println(s);
                    }
                });
        Disposable disposable2 = Observable
                .fromIterable(list)
                .subscribe(new Consumer<String>() {
                    @Override
                    public void accept(@NonNull String s) throws Exception {
                        System.out.println(s);
                    }
                });
    }

查看Disposable接口的源码,如下:

public interface Disposable {
        void dispose();
        boolean isDisposed();
}

其中isDisposed()方法用来判断当前订阅是否失效,dispose()方法用来取消当前订阅。
只有当观察者Observer与可观察对象Observable之间建立订阅关系,并且订阅关系有效时,Observer才能对Observable进行响应。如果Observer在响应Observable的过程中,订阅关系被取消,则Observer无法对取消订阅关系之后Observable的行为作出响应。
运行下面的代码,当Observable接收到第5条数据时,取消订阅关系。

public void demo8() {
        Observable.create(new ObservableOnSubscribe<Integer>() {
            @Override
            public void subscribe(ObservableEmitter<Integer> emitter) throws Exception {
                for (int i = 0; i < 10; i++) {
                    System.out.println("发送" + i);
                    emitter.onNext(i);
                }
                emitter.onComplete();
            }
        }).subscribe(new Observer<Integer>() {
            private Disposable disposable;

            @Override
            public void onSubscribe(Disposable d) {
                disposable = d;
            }

            @Override
            public void onNext(Integer integer) {
                System.out.println("接收" + integer);
                if (integer > 4) disposable.dispose();
            }

            @Override
            public void onError(Throwable e) {
                e.printStackTrace();
            }

            @Override
            public void onComplete() {
                System.out.println("数据接受完成");
            }
        });
    }

控制台日志如下:

 I/System.out: 发送0
 I/System.out: 接收0
 I/System.out: 发送1
 I/System.out: 接收1
 I/System.out: 发送2
 I/System.out: 接收2
 I/System.out: 发送3
 I/System.out: 接收3
 I/System.out: 发送4
 I/System.out: 接收4
 I/System.out: 发送5
 I/System.out: 接收5
 I/System.out: 发送6
 I/System.out: 发送7
 I/System.out: 发送8
 I/System.out: 发送9

可以发现取消订阅关系之前,Observable发送一条数据,Observe便接收一条,但是取消订阅关系之后,Observe将不再接收Observable发送的数据。

上一篇:Rxjava2入门教程一:函数响应式编程及概述
下一篇;Rxjava2入门教程三:Operators操作符

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 157,012评论 4 359
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 66,589评论 1 290
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 106,819评论 0 237
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,652评论 0 202
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 51,954评论 3 285
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,381评论 1 210
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,687评论 2 310
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,404评论 0 194
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,082评论 1 238
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,355评论 2 241
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 31,880评论 1 255
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,249评论 2 250
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 32,864评论 3 232
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,007评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,760评论 0 192
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,394评论 2 269
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,281评论 2 259

推荐阅读更多精彩内容