RxJava2.0 操作符(4)—— Combin 组合操作

这个页面展示的操作符可用于组合多个 Observables。

And/Then/When — 通过模式和计划组合多个 Observables 发射的数据集合。
CombineLatest — 当两个 Observables 中的任何一个发射了一个数据时,通过一个指定的函数组合每个 Observable 发射的最新数据(一共两个数据),然后发射这个函数的结果。
Join — 无论何时,如果一个 Observable 发射了一个数据项,只要在另一个 Observable 发射的数据项定义的时间窗口内,就将两个 Observable 发射的数据合并发射。
Merge — 将多个 Observable 合并为一个。
StartWith — 在数据序列的开头插入一条指定的项。
Switch — 将一个发射 Observables 的 Observable 转换成另一个 Observable,后者发射这些 Observables 最近发射的数据。
Zip — 使用一个函数组合多个 Observable 发射的数据集合,然后再发射这个结果。

4.1 And/Then/When

通过模式和计划组合多个 Observables 发射的数据集合。


and,then,when
and,then,when

示例代码:

4.2 CombineLatest

当两个 Observables 中的任何一个发射了一个数据时,通过一个指定的函数组合每个 Observable 发射的最新数据(一共两个数据),然后发射这个函数的结果。

CombineLatest
CombineLatest

CombineLatest 操作符行为类似于 zip,但是只有当原始的 Observable 中的每一个都发射了一条数据时 zip 才发射数据。CombineLatest 则在原始的 Observable 中任意一个发射了数据时发射一条数据。当原始 Observables 的任何一个发射了一条数据时,CombineLatest 使用一个函数结合它们最近发射的数据,然后发射这个函数的返回值。

实践:可用于填写表单提交,需要满足多个条件才触发“提交”按钮。

示例代码:

//2个有差值的observable
Observable<Long> ob1 = Observable.interval(100, TimeUnit.MILLISECONDS);
Observable<Long> ob2 = Observable.interval(150, TimeUnit.MILLISECONDS);

Observable.combineLatest(ob1, ob2, new BiFunction<Long, Long, String>() {
    @Override
    public String apply(@NonNull Long aLong, @NonNull Long aLong2) throws Exception {
        return  aLong + " - " + aLong2;
    }
}).take(6).subscribe(new Consumer<String>() {
    @Override
    public void accept(@NonNull String s) throws Exception {
        Log.e(TAG, "accept : " + s);
    }
});

输出结果:

accept : 0 - 0
accept : 1 - 0
accept : 1 - 1
accept : 2 - 1
accept : 3 - 1
accept : 3 - 2

4.3 Join

无论何时,如果一个 Observable 发射了一个数据项,只要在另一个 Observable 发射的数据项定义的时间窗口内,就将两个 Observable 发射的数据合并发射。

Join
Join

Join 操作符结合两个 Observable 发射的数据,基于时间窗口(你定义的针对每条数据特定的原则)选择待集合的数据项。你将这些时间窗口实现为一些 Observables,它们的生命周期从任何一条 Observable 发射的每一条数据开始。当这个定义时间窗口的 Observable 发射了一条数据或者完成时,与这条数据关联的窗口也会关闭。只要这条数据的窗口是打开的,它将继续结合其它 Observable 发射的任何数据项。你定义一个用于结合数据的函数。
示例代码:

Observable<Integer> ob1 = Observable.just(1, 2);
Observable<String> ob2 = Observable.just("A", "B", "C");
ob1.join(ob2, new Function<Integer, ObservableSource<String>>() {//ob1产生结果声明周期控制函数
        @Override
        public ObservableSource<String> apply(@NonNull Integer integer) throws Exception {
//              return Observable.just(integer + "00");
            return Observable.just(integer + "00").delay(100, TimeUnit.MILLISECONDS);
//              return Observable.just(integer + "00").delay(200, TimeUnit.MILLISECONDS);
//              return Observable.just(integer + "00").delay(300, TimeUnit.MILLISECONDS);
        }
    }, new Function<String, ObservableSource<String>>() {//ob2产生结果声明周期控制函数
        @Override
        public ObservableSource<String> apply(@NonNull String s) throws Exception {
            return Observable.just("Function1:" + s).delay(200, TimeUnit.MILLISECONDS);
        }
    }, new BiFunction<Integer, String, Object>() {//ob1 和ob2产生结果的合并规则
        @Override
        public Object apply(@NonNull Integer integer, @NonNull String s) throws Exception {
//              Thread.sleep(1000);
            return  "string = " + s + ", integer = " + integer;
        }
    }

).subscribe(new Consumer<Object>() {
    @Override
    public void accept(@NonNull Object o) throws Exception {
        Log.e(TAG, "accept : " + o.toString());
    }
});

输出结果:

accept : string = A, integer = 1
accept : string = A, integer = 2
accept : string = B, integer = 1
accept : string = B, integer = 2
accept : string = C, integer = 1
accept : string = C, integer = 2

Join 的效果类似于排列组合,把第一个数据源 ob1 作为基座窗口,它根据自己的节奏不断发射数据元素。第二个数据源 ob2 ,每发射一个数据,我们都把它和第一个数据源 ob1 中已经发射的数据进行一对一匹配;

如果某一时刻 ob2 发射了一个数据 B ,此时 ob1 已经发射了 1,2 共两个数据,那么我们的合并操作就会把 B 依次与 1,2 配对,得到两组数据: (1,B)、(2,B)

4.3.1 GroupJoin

GroupJoin
GroupJoin

GroupJoin 运算符与 Join 类似。将第两个 Observable 的发射放在了 GroupJoin 第三个参数中。具体看代码吧。

示例代码:

Observable<Integer> ob1 = Observable.just(1, 2);
Observable<String> ob2 = Observable.just("A", "B", "C");
ob1.groupJoin(ob2, new Function<Integer, ObservableSource<String>>() {//ob1产生结果声明周期控制函数
    @Override
    public ObservableSource<String> apply(@NonNull Integer integer) throws Exception {
        return Observable.just(integer + "00").delay(300, TimeUnit.MILLISECONDS);
    }
}, new Function<String, ObservableSource<String>>() {//ob2产生结果声明周期控制函数
    @Override
    public ObservableSource<String> apply(@NonNull String s) throws Exception {
        return Observable.just("Function1:" + s).delay(200, TimeUnit.MILLISECONDS);
    }
    //与 join 运算符的差别主要是在这里。
}, new BiFunction<Integer, Observable<String>, Observable<String>>() {
    @Override
    public Observable<String> apply(@NonNull final Integer i, @NonNull Observable<String> sob) throws Exception {
        return sob.map(new Function<String, String>() {
            @Override
            public String apply(@NonNull String s) throws Exception {
                return  "string = " + s + ", integer = " + i;
            }
        });
    }
}).subscribe(new Consumer<Observable<String>>() {
    @Override
    public void accept(@NonNull Observable<String> sob) throws Exception {
        sob.subscribe(new Consumer<String>() {
            @Override
            public void accept(@NonNull String s) throws Exception {
                Log.e(TAG, "accept : " + s);
            }
        });
    }
});

输出结果:

accept : string = A, integer = 2
accept : string = B, integer = 1
accept : string = B, integer = 2
accept : string = C, integer = 1
accept : string = C, integer = 2

4.4 Merge

将多个 Observable 合并为一个 Observable。

merge
merge

Merge 可能会让合并的 Observables 发射的数据交错(有一个类似的操作符 Concat 不会让数据交错,它会按顺序一个接着一个发射多个 Observable)。

示例代码:

Observable<String> ob1 = Observable.interval(100, TimeUnit.MILLISECONDS)
    .map(new Function<Long, String>() {
        @Override
        public String apply(@NonNull Long aLong) throws Exception {
            //强迫症与上图一致
            return "" + ((aLong + 1) * 20);
        }
    }).take(5);
Observable<String> ob2 = Observable.create(new ObservableOnSubscribe<String>() {
    @Override
    public void subscribe(@NonNull ObservableEmitter<String> emitter) throws Exception {
        Thread.sleep(350);
        emitter.onNext("1");
        Thread.sleep(250);
        emitter.onNext("1");
        emitter.onComplete();
    }
});
Observable.merge(ob1, ob2).subscribe(new Consumer<String>() {
    @Override
    public void accept(@NonNull String s) throws Exception {
        Log.e(TAG, "accept : " + s);
    }
});

输出结果:

accept : 20
accept : 40
accept : 60
accept : 1
accept : 80
accept : 100
accept : 1

4.5 StartWith

在数据序列的开头插入一条指定的项。

StartWith
StartWith

如果你想要一个 Observable 在发射数据之前先发射一个指定的数据序列,可以使用 StartWith 操作符。(如果你想一个 Observable 发射的数据末尾追加一个数据序列可以使用 Concat 操作符)。
示例代码:

Observable<Integer> ob1 = Observable.range(2, 3);
ob1.startWith(4).subscribe(new Consumer<Integer>() {
    @Override
    public void accept(@NonNull Integer integer) throws Exception {
        Log.e(TAG, "accept : " + integer);
    }
});

输出结果:

accept : 4
accept : 2
accept : 3
accept : 4

4.6 Switch

将一个发射 Observables 的 Observable 转换成另一个 Observable,后者发射这些 Observables 最近发射的数据。


Switch
Switch

Switch 订阅一个发射多个 Observables 的 Observable。它每次观察那些 Observables 中的一个,Switch 返回的这个 Observable 取消订阅前一个发射数据的 Observable,开始发射最近的 Observable 发射的数据。注意:当原始 Observable 发射了一个新的 Observable 时(不是这个新的 Observable 发射了一条数据时),它将取消订阅之前的那个 Observable。这意味着,在后来那个 Observable 产生之后到它开始发射数据之前的这段时间里,前一个 Observable 发射的数据将被丢弃(就像图例上的那个黄色圆圈一样)。
示例代码:

Observable.switchOnNext(Observable.interval(100, TimeUnit.MILLISECONDS)
        .map(new Function<Long, ObservableSource<String>>() {
            @Override
            public ObservableSource<String> apply(@NonNull final Long aL2) throws Exception {
    //          Log.e(TAG, "al2 =" + aL2);
                return Observable.interval(30, TimeUnit.MILLISECONDS)
                .map(new Function<Long, String>() {
                    @Override
                    public String apply(@NonNull Long aL3) throws Exception {
    //                  Log.e(TAG, "al3 =" + aL3);
                        return "al2 =" + aL2 + " , al3 = " + aL3;
                    }
                });
            }
    }))
    .take(6)
    .subscribe(new Consumer<String>() {
        @Override
        public void accept(@NonNull String s) throws Exception {
            Log.e(TAG, "accept : " + s);
        }
    });

输出结果:

accept : al2 =0 , al3 = 0
accept : al2 =0 , al3 = 1
accept : al2 =0 , al3 = 2
accept : al2 =1 , al3 = 0
accept : al2 =1 , al3 = 1
accept : al2 =2 , al3 = 0

4.7 Zip

使用一个函数组合多个 Observable 发射的数据集合,然后再发射这个结果。

Zip
Zip

Zip 操作符返回一个 Obversable,它使用这个函数按顺序结合两个或多个 Observables 发射的数据项,然后它发射这个函数返回的结果。它按照严格的顺序应用这个函数。它只发射与发射数据项最少的那个 Observable 一样多的数据。

RxJava 将这个操作符实现为 zip 和 zipWith。

4.7.1 Zip

zip
zip

示例代码:

Observable<Integer> ob1 = Observable.just(1, 2, 3, 4, 5);
Observable<String> ob2 = Observable.just("A", "B", "C", "D");
Observable.zip(ob1, ob2, new BiFunction<Integer, String, String>() {
    @Override
    public String apply(@NonNull Integer i, @NonNull String s) throws Exception {
        return i + " + " + s;
    }
}).subscribe(new Consumer<String>() {
    @Override
    public void accept(@NonNull String s) throws Exception {
        Log.e(TAG, "accept:" + s);
    }
});

输出结果:

accept:1 + A
accept:2 + B
accept:3 + C
accept:4 + D

3.7.2 ZipWith

zipWith
zipWith

示例代码:

Observable<Integer> ob1 = Observable.just(1, 2, 3, 4, 5);
Observable<String> ob2 = Observable.just("A", "B", "C", "D");
ob1.zipWith(ob2, new BiFunction<Integer, String, String>() {
    @Override
    public String apply(@NonNull Integer i, @NonNull String s) throws Exception {
        return i + "-" + s;
    }
}).subscribe(new Consumer<String>() {
    @Override
    public void accept(@NonNull String s) throws Exception {
        Log.e(TAG, "accept:" + s);
    }
});

输出结果:

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

推荐阅读更多精彩内容