操作符就是為了解決對(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ù)。