成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

RxJava操作符系列二(下)

移動開發 Android
操作符對原始Observable發射的第一項數據應用一個函數,然后將那個函數的結果作為自己的第一項數據發射。它將函數的結果同第二項數據一起填充給這個函數來產生它自己的第二項數據。它持續進行這個過程來產生剩余的數據序列。

[[180037]]

接上文

輸出日志信息

  1. call:2 ConcatMap RxNewThreadScheduler-5 
  2. onNext: ConcatMap 101 ConcatMap 
  3. call:2 ConcatMap RxNewThreadScheduler-6 
  4. onNext: ConcatMap 102 ConcatMap 
  5. call:2 ConcatMap RxNewThreadScheduler-7 
  6. onNext: ConcatMap 103 ConcatMap 
  7. onCompleted: ConcatMap  

通過該操作符和flatMap輸出的日志信息,很容易看出flatMap并沒有保證數據源的順序性,但是ConcatMap操作符保證了數據源的順序性。在應用中,如果你對數據的順序性有要求的話,就需要使用ConcatMap。若沒有要求,二者皆可使用。

SwitchMap

當原始Observable發射一個新的數據(Observable)時,它將取消訂閱并停止監視產生執之前那個數據的Observable,只監視當前這一個.

  1. Integer[] integers = {1, 2, 3}; 
  2. Observable.from(integers).switchMap(new Func1>() { 
  3.             @Override 
  4.             public Observable call(Integer integer) { 
  5.                 Log.e(TAG, "call: SwitchMap" + Thread.currentThread().getName()); 
  6.                 //如果不通過subscribeOn(Schedulers.newThread())在在子線程模擬并發操作,所有數據源依然會全部輸出,也就是并發操作此操作符才有作用 
  7.                 //若在此通過Thread。sleep()設置等待時間,則輸出信息會不一樣。相當于模擬并發程度 
  8.                 return Observable.just((integer + 100) + "SwitchMap").subscribeOn(Schedulers.newThread()); 
  9.             } 
  10.         }).observeOn(AndroidSchedulers.mainThread()).subscribe(new Subscriber() { 
  11.             @Override 
  12.             public void onCompleted() { 
  13.                 Log.e(TAG, "onCompleted: SwitchMap"); 
  14.             } 
  15.   
  16.             @Override 
  17.             public void onError(Throwable e) { 
  18.                 Log.e(TAG, "onError: SwitchMap"); 
  19.             } 
  20.   
  21.             @Override 
  22.             public void onNext(String s) { 
  23.                 Log.e(TAG, "onNext: SwitchMap "+s); 
  24.             } 
  25.         });  

輸出日志信息

  1. call: SwitchMapmain 
  2. call: SwitchMapmain 
  3. call: SwitchMapmain 
  4. onNext: SwitchMap 106SwitchMap 
  5. onCompleted: SwitchMap  

當數據源較多時,并不一定是只輸出***一項數據,有可能輸出幾項數據,也可能是全部。

GroupBy

看到這個詞你就應該想到了這個操作符的作用,就是你理解的含義,他將數據源按照你的約定進行分組。我們通過groupBy實行將1到10的數據進行就劃分,代碼如下 

  1. Observable.range(1, 10).groupBy(new Func1() { 
  2.             @Override 
  3.             public Boolean call(Integer integer) { 
  4.                 return integer % 2 == 0; 
  5.             } 
  6.         }).subscribe(new Subscriber>() { 
  7.             @Override 
  8.             public void onCompleted() { 
  9.                 Log.e(TAG, "onCompleted:1 "); 
  10.             } 
  11.   
  12.             @Override 
  13.             public void onError(Throwable e) { 
  14.                 Log.e(TAG, "onError:1 "); 
  15.             } 
  16.   
  17.             @Override 
  18.             public void onNext(GroupedObservable booleanIntegerGroupedObservable) { 
  19.                 booleanIntegerGroupedObservable.toList().subscribe(new Subscriber>() { 
  20.                     @Override 
  21.                     public void onCompleted() { 
  22.                         Log.e(TAG, "onCompleted:2 " ); 
  23.                     } 
  24.   
  25.                     @Override 
  26.                     public void onError(Throwable e) { 
  27.                         Log.e(TAG, "onError:2 "); 
  28.                     } 
  29.   
  30.                     @Override 
  31.                     public void onNext(List integers) { 
  32.                         Log.e(TAG, "onNext:2 "+integers); 
  33.                     } 
  34.                 }); 
  35.             } 
  36.         });  

輸出日志信息

  1. onNext:2 [1, 3, 5, 7, 9] 
  2. onCompleted:2 
  3. onNext:2 [2, 4, 6, 8, 10] 
  4. onCompleted:2 
  5. onCompleted:1  

在上面代碼中booleanIntegerGroupedObservable變量有一個getKey()方法,該方法返回的是分組的key,他的值就是groupBy方法call回調所用函數的值,在上面也就是integer % 2 == 0的值,及true和false。有幾個分組也是有此值決定的。

Scan

操作符對原始Observable發射的***項數據應用一個函數,然后將那個函數的結果作為自己的***項數據發射。它將函數的結果同第二項數據一起填充給這個函數來產生它自己的第二項數據。它持續進行這個過程來產生剩余的數據序列。

例如計算1+2+3+4的和

  1. Observable.range(1,4).scan(new Func2() { 
  2.             @Override 
  3.             public Integer call(Integer integerInteger integer2) { 
  4.                 Log.e(TAG, "call: integer:"+integer+"  integer2 "+integer2); 
  5.                 return integer+integer2; 
  6.             } 
  7.         }).subscribe(new Subscriber() { 
  8.             @Override 
  9.             public void onCompleted() { 
  10.                 Log.e(TAG, "onCompleted: "); 
  11.             } 
  12.   
  13.             @Override 
  14.             public void onError(Throwable e) { 
  15.                 Log.e(TAG, "onError: " ); 
  16.             } 
  17.   
  18.             @Override 
  19.             public void onNext(Integer integer) { 
  20.                 Log.e(TAG, "onNext: "+integer ); 
  21.             } 
  22.         });  

輸出日志信息 

  1. onNext: 1 
  2. call: integer:1  integer2 2 
  3. onNext: 3 
  4. call: integer:3  integer2 3 
  5. onNext: 6 
  6. call: integer:6  integer2 4 
  7. onNext: 10 
  8. onCompleted:  

對于scan有一個重載方法,可以設置一個初始值,如上面代碼,初始值設置為10,只需將scan加個參數scan(10,new Func2)。

Buffer

操作符將一個Observable變換為另一個,原來的Observable正常發射數據,變換產生的Observable發射這些數據的緩存集合,如果原來的Observable發射了一個onError通知,Buffer會立即傳遞這個通知,而不是首先發射緩存的數據,即使在這之前緩存中包含了原始Observable發射的數據。

示例代碼

  1. Observable.range(10, 6).buffer(2).subscribe(new Subscriber>() { 
  2.             @Override 
  3.             public void onCompleted() { 
  4.                 Log.e(TAG, "onCompleted: "); 
  5.             } 
  6.   
  7.             @Override 
  8.             public void onError(Throwable e) { 
  9.                 Log.e(TAG, "onError: "); 
  10.             } 
  11.   
  12.             @Override 
  13.             public void onNext(List integers) { 
  14.                 Log.e(TAG, "onNext: " + integers); 
  15.             } 
  16.         });  

輸出日志信息

  1. onNext: [10, 11] 
  2. onNext: [12, 13] 
  3. onNext: [14, 15] 
  4. onCompleted:  

上面一次性訂閱兩個數據,如果設置參數為6,就一次性訂閱。buffer的另一重載方法buffer(count, skip)從原始Observable的***項數據開始創建新的緩存(長度count),此后每當收到skip項數據,用count項數據填充緩存:開頭的一項和后續的count-1項,它以列表(List)的形式發射緩存,取決于count和skip的值,這些緩存可能會有重疊部分(比如skip count時)。具體執行結果,你可以設置不同的skip和count觀察輸出日志,查看執行結果及流程。

Window

Window和Buffer類似,但不是發射來自原始Observable的數據包,它發射的是Observables,這些Observables中的每一個都發射原始Observable數據的一個子集,***發射一個onCompleted通知。

  1. Observable.range(10, 6).window(2).subscribe(new Subscriber>() { 
  2.             @Override 
  3.             public void onCompleted() { 
  4.                 Log.e(TAG, "onCompleted1: "); 
  5.             } 
  6.   
  7.             @Override 
  8.             public void onError(Throwable e) { 
  9.                 Log.e(TAG, "onError1: "); 
  10.             } 
  11.   
  12.             @Override 
  13.             public void onNext(Observable integerObservable) { 
  14.                 Log.e(TAG, "onNext1: "); 
  15.                 tv1.append("\n"); 
  16.                 integerObservable.subscribe(new Subscriber() { 
  17.                     @Override 
  18.                     public void onCompleted() { 
  19.                         Log.e(TAG, "onCompleted2: "); 
  20.                     } 
  21.   
  22.                     @Override 
  23.                     public void onError(Throwable e) { 
  24.                         Log.e(TAG, "onError2: "); 
  25.                     } 
  26.   
  27.                     @Override 
  28.                     public void onNext(Integer integer) { 
  29.                         Log.e(TAG, "onNext2: "+integer); 
  30.                     } 
  31.                 }); 
  32.             } 
  33.         });  

輸出日志信息

  1. onNext2: 10 
  2. onNext2: 11 
  3. onCompleted2: 
  4. onNext2: 12 
  5. onNext2: 13 
  6. onCompleted2: 
  7. onNext2: 14 
  8. onNext2: 15 
  9. onCompleted2: 
  10. onCompleted1:  

window和buffer一樣也有不同的重載方法。這兩個操作符相對其他操作符不太容易理解,可以去RxJava GitHub理解,里面有圖示解析。當然***的理解方式就是通過更改變量的值,去觀察輸出的日志信息。

好了,這篇文章就介紹到這里。若文中有錯誤的地方,歡迎指正。謝謝。

責任編輯:龐桂玉 來源: 安卓開發精選
相關推薦

2017-01-03 16:12:13

RxJava操作符Android

2016-12-28 09:48:09

AndroidRxJava操作符

2017-01-03 15:56:20

RxJava操作符Android

2016-11-17 15:35:51

RxJava操作Subscriber

2021-10-31 18:59:55

Python操作符用法

2010-07-14 14:55:07

Perl操作符

2009-08-19 17:26:28

C# 操作符

2011-04-08 16:26:14

JavaScript

2010-07-14 14:30:31

Perl操作符

2010-07-19 11:00:24

Perl操作符

2009-09-15 17:16:58

LINQ查詢操作符

2012-02-06 09:13:23

LINQ

2009-09-16 09:09:23

Linq Contai

2010-07-14 14:18:51

Perl操作符

2009-07-21 09:31:00

Scala操作符

2010-01-28 11:16:28

C++操作符

2009-07-14 18:34:22

Jython操作符重載

2010-07-13 11:11:39

Perl標量

2009-11-30 16:48:08

PHP操作符

2010-01-27 11:00:17

C++操作符
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 成人精品一区亚洲午夜久久久 | 亚洲成人av在线播放 | 欧美一区二区三区,视频 | 四虎午夜剧场 | 精品国产区 | 国产一区在线免费 | 亚洲欧美日本在线 | 欧美电影一区 | 久久综合狠狠综合久久综合88 | 免费一级欧美在线观看视频 | 日韩av一区二区在线观看 | 成人在线免费观看av | 国产日韩欧美精品一区二区三区 | 美国一级黄色片 | 在线播放中文字幕 | 日韩视频在线播放 | 欧美成人精品一区 | 不卡的av电影 | av男人天堂影院 | 91青青草视频 | 天天射色综合 | 午夜久久久 | 欧美成年视频 | 国产精品久久久久久亚洲调教 | 亚洲精品一区二区三区在线 | 色综合成人网 | 成人欧美一区二区三区1314 | 中文字幕1区| 亚洲精品在线看 | 天天草天天干天天 | 日韩精品一区二区三区在线观看 | 精品99久久久久久 | 黄色片免费在线观看 | 日韩高清中文字幕 | 久久综合九色综合欧美狠狠 | 欧美影院 | 91精品久久久久久久 | 一级大黄色片 | 一级做a爰片久久毛片 | 亚洲欧美日韩中文在线 | 国产真实精品久久二三区 |