美文网首页
RxJava学习之结合(组合)型操作符

RxJava学习之结合(组合)型操作符

作者: 菜鸟_一枚 | 来源:发表于2017-03-10 21:14 被阅读110次

    下面展示了可用于多个Observables的各种操作符

    • startWith()---在数据序列的开头增加一项数据
    • merge()---将多个Observable合并为一个
    • mergeDelayError()---合并多个Observables,让没有错误的Observable都完成后再发射错误通知
    • zip()---使用一个函数组合多个Observable发射的数据集合,然后在发射这个结果
    • and(),then(),and when()---(rxjava-joins)通过模式和计划组合多个Observables发射的数据集合
    • combineLatest()---当两个Observables中的任何一个发射了一个数据时,通过一个指定的函数组合每个Observable发射的最新数据(一共两个数据),然后发射这个函数的结果。
    • join() and groupJoin()---无论何时,如果一个Observable发射了一个数据项,只要在另一个Observable发射的数据项定义的时间窗口内,就将两个Observable发射的数据合并发射。
    • switchOnNext()---将一个发射Observables的Observable转换成另一个Observable,后者发射这些Observables最近发射的数据。

    StartWith

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



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



    你也可以传递一个Observable给startWith,它会将那个Observable的发射物插在原始Observable发射的数据序列之前,然后把这个当做自己的发射物集合。这可以看作是Concat的反转。

    /**
         * startWith操作符 解释:在源Observable输出之前插入指定数据项
         */
        private static void test() {
            Observable.just(7, 8, 9).startWith(11, 12)
            .subscribe(new Observer<Integer>() {
    
                @Override
                public void onCompleted() {
                    System.out.println("onCompleted ");
                }
    
                @Override
                public void onError(Throwable e) {
                    System.out.println("onError " + e.toString());
                }
    
                @Override
                public void onNext(Integer t) {
                    System.out.println("onNext t = " + t);
                }
            });
    
        }
    

    Merge操作符

    合并多个Observables的发射物



    使用Merge操作符你可以将多个Observables的输出合并,就好像它们是一个单个的Observable一样。

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

    正如图例上展示的,任何一个原始Observable的onError通知会被立即传递给观察者,而且会终止合并后的Observable。


    在很多ReactiveX实现中还有一个叫MergeDelayError的操作符,它的行为有一点不同,它会保留onError通知直到合并后的Observable所有的数据发射完成,在那时它才会把onError传递给观察者。

    RxJava将它实现为merge, mergeWith和mergeDelayError。


    /**
         * Merge 解释:将2-9个Observables合并到一个Observable中进行发射,合并后的数据可能会是交错(无序)的(如果想要没有交错,
         * 可以使用concat操作符) merge还可以传递一个Observable列表List,数组
         * 甚至是一个发射Observable序列的Observable,merge将合并它们的输出作为单个Observable的输出
         */
        private static void test1() {
            Observable<String> letterObservable = Observable.just("A", "B", "C", "D", "E", "F", "G", "H");
            Observable<Integer> numberObservable = Observable.just(1, 2, 3, 4, 5);
    
            Observable.merge(letterObservable, numberObservable)
            .subscribe(new Action1<Serializable>() {
    
                @Override
                public void call(Serializable value) {
                    System.out.println("value = " + value);
                }
            });
        }
    

    Zip操作符

    通过一个函数将多个Observables的发射物结合到一起,基于这个函数的结果为每个结合体发射单个数据项。


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

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


    zip的最后一个参数接受每个Observable发射的一项数据,返回被压缩后的数据,它可以接受一到九个参数:一个Observable序列,或者一些发射Observable的Observables。

    zipWith


    zipWith操作符总是接受两个参数,第一个参数是一个Observable或者一个Iterable。

    zip和zipWith默认不在任何特定的操作符上执行。

    /**
         * Zip 
         * 结合两个或多个Observables发射的数据项,每个数据只能组合一次,而且都是有序的。
         * 它只发射与发射数据项最少的那个Observable一样多的数据。
         * 
         * 应用场景参考:http://blog.csdn.net/jdsjlzx/article/details/51724087
         */
        private static void test2() {
            Observable<String> letterObservable = Observable.just("A", "B", "C", "D", "E", "F", "G", "H");
            Observable<Integer> numberObservable = Observable.just(1, 2, 3, 4, 5);
    
            Observable.zip(letterObservable, numberObservable, new Func2<String, Integer, String>() {
    
                @Override
                public String call(String t1, Integer t2) {
                    return t1 + t2;
                }
            }).subscribe(new Action1<String>() {
    
                @Override
                public void call(String value) {
                    System.out.println("value = " + value);
                }
            });
        }
    

    CombineLatest操作符

    当两个Observables中的任何一个发射了数据时,使用一个函数结合每个Observable发射的最近数据项,并且基于这个函数的结果发射数据。



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



    RxJava将这个操作符实现为combineLatest,它接受二到九个Observable作为参数,或者单个Observables列表作为参数。它默认不在任何特定的调度器上执行。
    /**
         * CombineLatest
         * 解释:combineLatest操作符把两个Observable产生的结果进行合并,合并的结果组成一个新的Observable。
         * 这两个Observable中任意一个Observable产生的结果,都和另一个Observable最后产生的结果,按照一定的规则进行合并。
         */
        private static void test3() {
            Observable<String> letterObservable = Observable.just("A", "B", "C");
            Observable<Integer> numberObservable = Observable.just(4, 5);
    
            //letterObservable numberObservable谁在前谁在后都会对执行结果会有影响
            Observable.combineLatest( numberObservable, letterObservable,new Func2<Integer, String,  String>() {
    
                @Override
                public String call(Integer t1, String t2) {
                    System.out.println("combine  t1 = " + t1 + " | t2 = " + t2);// t1的值一开始就是letterObservable中的最后一个值
                    return t1 + t2;
                }
            }).subscribe(new Action1<String>() {
    
                @Override
                public void call(String value) {
                    System.out.println("value = " + value);
                }
            });
        }
    

    withLatestFrom操作符


    withLatestFrom操作符还在开发中,不是1.0版本的一部分。类似于combineLatest,但是只在单个原始Observable发射了一条数据时才发射数据。

    Join操作符

    任何时候,只要在另一个Observable发射的数据定义的时间窗口内,这个Observable发射了一条数据,就结合两个Observable发射的数据。



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


    
        /**
         * Join 类似于combineLatest操作符,但是join操作符可以控制每个Observable产生结果的生命周期,在每个结果的生命周期内,
         * 可以与另一个Observable产 生的结果按照一定的规则进行合并
         * 
         * Join(Observable,Func1,Func1,Func2) 需要传递四个参数
         * 
         * join操作符的用法如下: observableA.join(observableB, observableA产生结果生命周期控制函数,
         * observableB产生结果生命周期控制函数, observableA产生的结果与observableB产生的结果的合并规则)
         * 
         * 一句话概括:在observableA的生命周期内:observableB输出的数据项与observableA输出的数据项每个合并
         * 
         * test4() 没有任何合并结果输出 分析:同一线程:observableA的生命周期已经执行完了,observableB还没出来,所以合并不了
         * 
         */
        private static void test4() {
    
            Observable<Integer> observableA = Observable.range(1, 5);
    
            List<Integer> data = Arrays.asList(6, 7, 8, 9, 10);
            Observable<Integer> observableB = Observable.from(data);
    
            observableA.join(observableB, new Func1<Integer, Observable<Integer>>() {
                @Override
                public Observable<Integer> call(Integer value) {
                    //return Observable.just(value);
                    return Observable.just(value).delay(1, TimeUnit.SECONDS);
                }
            }, new Func1<Integer, Observable<Integer>>() {
                @Override
                public Observable<Integer> call(Integer value) {
                    return Observable.just(value);
                }
            }, new Func2<Integer, Integer, Integer>() {
                @Override
                public Integer call(Integer value1, Integer value2) {
                    System.out.println("left: " + value1 + "  right:" + value2);
                    return value1 + value2;
                }
            }).subscribe(new Observer<Integer>() {
                @Override
                public void onCompleted() {
                    System.out.println("onCompleted");
                }
    
                @Override
                public void onError(Throwable e) {
                }
    
                @Override
                public void onNext(Integer value) {
                    System.out.println("onNext value = " + value);
                }
            });
    
        }
    

    相关文章

      网友评论

          本文标题:RxJava学习之结合(组合)型操作符

          本文链接:https://www.haomeiwen.com/subject/mhvogttx.html