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

2017-03-10  本文已影响110人  菜鸟_一枚

下面展示了可用于多个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);
            }
        });

    }
上一篇下一篇

猜你喜欢

热点阅读