一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

服務(wù)器之家:專注于服務(wù)器技術(shù)及軟件下載分享
分類導(dǎo)航

PHP教程|ASP.NET教程|JAVA教程|ASP教程|

服務(wù)器之家 - 編程語言 - JAVA教程 - Java的RxJava庫操作符的用法及實(shí)例講解

Java的RxJava庫操作符的用法及實(shí)例講解

2020-05-18 12:17hi大頭鬼hi JAVA教程

RxJava由于提供異步和基于事件的支持在Android開發(fā)者中獲得了不少人氣,這里我們就來看一下Java的RxJava庫操作符的用法及實(shí)例講解,需要的朋友可以參考下

操作符就是為了解決對(duì)Observable對(duì)象的變換的問題,操作符用于在Observable和最終的Subscriber之間修改Observable發(fā)出的事件。RxJava提供了很多很有用的操作符。
比如map操作符,就是用來把把一個(gè)事件轉(zhuǎn)換為另一個(gè)事件的。
 

?
1
2
3
4
5
6
7
8
Observable.just("Hello, world!")
 .map(new Func1<String, String>() {
   @Override
   public String call(String s) {
     return s + " -Dan";
   }
 })
 .subscribe(s -> System.out.println(s));

使用lambda可以簡(jiǎn)化為

?
1
2
3
Observable.just("Hello, world!")
  .map(s -> s + " -Dan")
  .subscribe(s -> System.out.println(s));

是不是很酷?map()操作符就是用于變換Observable對(duì)象的,map操作符返回一個(gè)Observable對(duì)象,這樣就可以實(shí)現(xiàn)鏈?zhǔn)秸{(diào)用,在一個(gè)Observable對(duì)象上多次使用map操作符,最終將最簡(jiǎn)潔的數(shù)據(jù)傳遞給Subscriber對(duì)象。


map操作符進(jìn)階
map操作符更有趣的一點(diǎn)是它不必返回Observable對(duì)象返回的類型,你可以使用map操作符返回一個(gè)發(fā)出新的數(shù)據(jù)類型的observable對(duì)象。
比如上面的例子中,subscriber并不關(guān)心返回的字符串,而是想要字符串的hash值
 

?
1
2
3
4
5
6
7
8
Observable.just("Hello, world!")
  .map(new Func1<String, Integer>() {
    @Override
    public Integer call(String s) {
      return s.hashCode();
    }
  })
  .subscribe(i -> System.out.println(Integer.toString(i)));

很有趣吧?我們初始的Observable返回的是字符串,最終的Subscriber收到的卻是Integer,當(dāng)然使用lambda可以進(jìn)一步簡(jiǎn)化代碼:
 

?
1
2
3
Observable.just("Hello, world!")
  .map(s -> s.hashCode())
  .subscribe(i -> System.out.println(Integer.toString(i)));

前面說過,Subscriber做的事情越少越好,我們?cè)僭黾右粋€(gè)map操作符
 

?
1
2
3
4
Observable.just("Hello, world!")
  .map(s -> s.hashCode())
  .map(i -> Integer.toString(i))
  .subscribe(s -> System.out.println(s));

不服?
是不是覺得我們的例子太簡(jiǎn)單,不足以說服你?你需要明白下面的兩點(diǎn):
1.Observable和Subscriber可以做任何事情
Observable可以是一個(gè)數(shù)據(jù)庫查詢,Subscriber用來顯示查詢結(jié)果;Observable可以是屏幕上的點(diǎn)擊事件,Subscriber用來響應(yīng)點(diǎn)擊事件;Observable可以是一個(gè)網(wǎng)絡(luò)請(qǐng)求,Subscriber用來顯示請(qǐng)求結(jié)果。
2.Observable和Subscriber是獨(dú)立于中間的變換過程的。
在Observable和Subscriber中間可以增減任何數(shù)量的map。整個(gè)系統(tǒng)是高度可組合的,操作數(shù)據(jù)是一個(gè)很簡(jiǎn)單的過程。

實(shí)例
1.準(zhǔn)備工作
假設(shè)我有這樣一個(gè)方法:
這個(gè)方法根據(jù)輸入的字符串返回一個(gè)網(wǎng)站的url列表(啊哈,搜索引擎)
 

?
1
Observable<List<String>> query(String text); 

現(xiàn)在我希望構(gòu)建一個(gè)健壯系統(tǒng),它可以查詢字符串并且顯示結(jié)果。根據(jù)上一篇blog的內(nèi)容,我們可能會(huì)寫出下面的代碼:
 

?
1
2
3
4
5
6
query("Hello, world!")
  .subscribe(urls -> {
    for (String url : urls) {
      System.out.println(url);
    }
  });

這種代碼當(dāng)然是不能容忍的,因?yàn)樯厦娴拇a使我們喪失了變化數(shù)據(jù)流的能力。一旦我們想要更改每一個(gè)URL,只能在Subscriber中來做。我們竟然沒有使用如此酷的map()操作符!!!

當(dāng)然,我可以使用map操作符,map的輸入是urls列表,處理的時(shí)候還是要for each遍歷,一樣很蛋疼。

萬幸,還有Observable.from()方法,它接收一個(gè)集合作為輸入,然后每次輸出一個(gè)元素給subscriber:
 

?
1
2
Observable.from("url1", "url2", "url3")
  .subscribe(url -> System.out.println(url));

我們來把這個(gè)方法使用到剛才的場(chǎng)景: 

?
1
2
3
4
5
query("Hello, world!")
  .subscribe(urls -> {
    Observable.from(urls)
      .subscribe(url -> System.out.println(url));
  });


雖然去掉了for each循環(huán),但是代碼依然看起來很亂。多個(gè)嵌套的subscription不僅看起來很丑,難以修改,更嚴(yán)重的是它會(huì)破壞某些我們現(xiàn)在還沒有講到的RxJava的特性。

2.改進(jìn)
救星來了,他就是flatMap()。
Observable.flatMap()接收一個(gè)Observable的輸出作為輸入,同時(shí)輸出另外一個(gè)Observable。直接看代碼:

?
1
2
3
4
5
6
7
8
query("Hello, world!")
  .flatMap(new Func1<List<String>, Observable<String>>() {
    @Override
    public Observable<String> call(List<String> urls) {
      return Observable.from(urls);
    }
  })
  .subscribe(url -> System.out.println(url));

這里我貼出了整個(gè)的函數(shù)代碼,以方便你了解發(fā)生了什么,使用lambda可以大大簡(jiǎn)化代碼長(zhǎng)度:

?
1
2
3
query("Hello, world!")
 .flatMap(urls -> Observable.from(urls))
 .subscribe(url -> System.out.println(url));

flatMap()是不是看起來很奇怪?為什么它要返回另外一個(gè)Observable呢?理解flatMap的關(guān)鍵點(diǎn)在于,flatMap輸出的新的Observable正是我們?cè)赟ubscriber想要接收的。現(xiàn)在Subscriber不再收到List<String>,而是收到一些列單個(gè)的字符串,就像Observable.from()的輸出一樣。

這部分也是我當(dāng)初學(xué)RxJava的時(shí)候最難理解的部分,一旦我突然領(lǐng)悟了,RxJava的很多疑問也就一并解決了。

3.還可以更好
flatMap()實(shí)在不能更贊了,它可以返回任何它想返回的Observable對(duì)象。
比如下面的方法: 

?
1
2
// 返回網(wǎng)站的標(biāo)題,如果404了就返回null
Observable<String> getTitle(String URL);


接著前面的例子,現(xiàn)在我不想打印URL了,而是要打印收到的每個(gè)網(wǎng)站的標(biāo)題。問題來了,我的方法每次只能傳入一個(gè)URL,并且返回值不是一個(gè)String,而是一個(gè)輸出String的Observabl對(duì)象。使用flatMap()可以簡(jiǎn)單的解決這個(gè)問題。 

?
1
2
3
4
5
6
7
8
9
query("Hello, world!")
  .flatMap(urls -> Observable.from(urls))
  .flatMap(new Func1<String, Observable<String>>() {
    @Override
    public Observable<String> call(String url) {
      return getTitle(url);
    }
  })
  .subscribe(title -> System.out.println(title));

4.使用lambda:

?
1
2
3
4
query("Hello, world!")
  .flatMap(urls -> Observable.from(urls))
  .flatMap(url -> getTitle(url))
  .subscribe(title -> System.out.println(title));

是不是感覺很不可思議?我竟然能將多個(gè)獨(dú)立的返回Observable對(duì)象的方法組合在一起!帥呆了!
不止這些,我還將兩個(gè)API的調(diào)用組合到一個(gè)鏈?zhǔn)秸{(diào)用中了。我們可以將任意多個(gè)API調(diào)用鏈接起來。大家應(yīng)該都應(yīng)該知道同步所有的API調(diào)用,然后將所有API調(diào)用的回調(diào)結(jié)果組合成需要展示的數(shù)據(jù)是一件多么蛋疼的事情。這里我們成功的避免了callback hell(多層嵌套的回調(diào),導(dǎo)致代碼難以閱讀維護(hù))。現(xiàn)在所有的邏輯都包裝成了這種簡(jiǎn)單的響應(yīng)式調(diào)用。

5.豐富的操作符
目前為止,我們已經(jīng)接觸了兩個(gè)操作符,RxJava中還有更多的操作符,那么我們?nèi)绾问褂闷渌牟僮鞣麃砀倪M(jìn)我們的代碼呢?
getTitle()返回null如果url不存在。我們不想輸出"null",那么我們可以從返回的title列表中過濾掉null值!

?
1
2
3
4
5
query("Hello, world!")
  .flatMap(urls -> Observable.from(urls))
  .flatMap(url -> getTitle(url))
  .filter(title -> title != null)
  .subscribe(title -> System.out.println(title));

filter()輸出和輸入相同的元素,并且會(huì)過濾掉那些不滿足檢查條件的。

如果我們只想要最多5個(gè)結(jié)果:
 

?
1
2
3
4
5
6
query("Hello, world!")
  .flatMap(urls -> Observable.from(urls))
  .flatMap(url -> getTitle(url))
  .filter(title -> title != null)
  .take(5)
  .subscribe(title -> System.out.println(title));

take()輸出最多指定數(shù)量的結(jié)果。

如果我們想在打印之前,把每個(gè)標(biāo)題保存到磁盤: 

?
1
2
3
4
5
6
7
query("Hello, world!")
  .flatMap(urls -> Observable.from(urls))
  .flatMap(url -> getTitle(url))
  .filter(title -> title != null)
  .take(5)
  .doOnNext(title -> saveTitle(title))
  .subscribe(title -> System.out.println(title));

doOnNext()允許我們?cè)诿看屋敵鲆粋€(gè)元素之前做一些額外的事情,比如這里的保存標(biāo)題。

看到這里操作數(shù)據(jù)流是多么簡(jiǎn)單了么。你可以添加任意多的操作,并且不會(huì)搞亂你的代碼。

RxJava包含了大量的操作符。操作符的數(shù)量是有點(diǎn)嚇人,但是很值得你去挨個(gè)看一下,這樣你可以知道有哪些操作符可以使用。弄懂這些操作符可能會(huì)花一些時(shí)間,但是一旦弄懂了,你就完全掌握了RxJava的威力。

你甚至可以編寫自定義的操作符!這篇blog不打算將自定義操作符,如果你想的話,清自行Google吧。

感覺如何?
好吧,你是一個(gè)懷疑主義者,并且還很難被說服,那為什么你要關(guān)心這些操作符呢?

因?yàn)椴僮鞣梢宰屇銓?duì)數(shù)據(jù)流做任何操作。

將一系列的操作符鏈接起來就可以完成復(fù)雜的邏輯。代碼被分解成一系列可以組合的片段。這就是響應(yīng)式函數(shù)編程的魅力。用的越多,就會(huì)越多的改變你的編程思維。

另外,RxJava也使我們處理數(shù)據(jù)的方式變得更簡(jiǎn)單。在最后一個(gè)例子里,我們調(diào)用了兩個(gè)API,對(duì)API返回的數(shù)據(jù)進(jìn)行了處理,然后保存到磁盤。但是我們的Subscriber并不知道這些,它只是認(rèn)為自己在接收一個(gè)Observable<String>對(duì)象。良好的封裝性也帶來了編碼的便利!


在第三部分中,我將介紹RxJava的另外一些很酷的特性,比如錯(cuò)誤處理和并發(fā),這些特性并不會(huì)直接用來處理數(shù)據(jù)。

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 娇妻与老头绿文小说系列 | 厨房play黄瓜进去小说h | 5g影院天天影院天天爽影院网站 | 亚洲国产成人99精品激情在线 | 别停好爽好深好大好舒服视频 | 国产欧美久久一区二区 | 婷婷综合久久 | 黄网在线观看免费网站台湾swag | 99re思思| 久久中文字幕免费高清 | 国产成人精品福利色多多 | 国产一卡二卡四卡免费 | h肉动漫在线视频无修无遮挡 | 亚洲欧美综合一区 | 精品精品国产自在久久高清 | 精品91自产拍在线观看99re | 含羞草传媒网站免费进入欢迎 | www.尤物在线| 国产第一综合另类色区奇米 | 免费港剧在线观看港剧 | 扒开老女人 | 日韩成人影视 | 欧美日韩一区二区三区免费 | 五月色天在线视频综合观看 | 九九大香尹人视频免费 | 激情自拍网 | 久久电影午夜 | 亚洲春色综合另类网蜜桃 | 全黄h全肉细节文在线观看 全彩成人18h漫画 | 欧美一区二区三区免费看 | 日日免费视频 | 14一15sexvideo日本 | 国内久久精品视频 | 98在线视频噜噜噜国产 | 日韩免费在线看 | 卫生间被教官做好爽HH视频 | 欧美穿高跟鞋做爰 | 久久爽狠狠添AV激情五月 | 国产草逼视频 | 刺客女仆 | 天天天天天天天操 |