程序员人生 网站导航

深入浅出RxJava(二:操作符)

栏目:综合技术时间:2015-03-11 08:16:50
在第1篇blog中,我介绍了RxJava的1些基础知识,同时也介绍了map()操作符。固然如果你并没成心愿去使用RxJava我1点都不惊讶,毕竟才接触了这么点。看完这篇blog,我相信你肯定想立即在你的项目中使用RxJava了,这篇blog将介绍许多RxJava中的操作符,RxJava的强大性就来自于它所定义的操作符。

首先先看1个例子:

准备工作

假定我有这样1个方法:
这个方法根据输入的字符串返回1个网站的url列表(啊哈,搜索引擎)
Observable<List<String>> query(String text);
现在我希望构建1个硬朗系统,它可以查询字符串并且显示结果。根据上1篇blog的内容,我们可能会写出下面的代码:
query("Hello, world!") .subscribe(urls -> { for (String url : urls) { System.out.println(url); } });
这类代码固然是不能容忍的,由于上面的代码使我们丧失了变化数据流的能力。1旦我们想要更改每个URL,只能在Subscriber中来做。我们居然没有使用如此酷的map()操作符!!!

固然,我可使用map操作符,map的输入是urls列表,处理的时候还是要for each遍历,1样很蛋疼。

万幸,还有Observable.from()方法,它接收1个集合作为输入,然后每次输出1个元素给subscriber:
Observable.from("url1", "url2", "url3") .subscribe(url -> System.out.println(url));
我们来把这个方法使用到刚才的场景:
query("Hello, world!") .subscribe(urls -> { Observable.from(urls) .subscribe(url -> System.out.println(url)); });
虽然去掉了for each循环,但是代码仍然看起来很乱。多个嵌套的subscription不但看起来很丑,难以修改,更严重的是它会破坏某些我们现在还没有讲到的RxJava的特性。

改进

救星来了,他就是flatMap()。
Observable.flatMap()接收1个Observable的输出作为输入,同时输出另外1个Observable。直接看代码:
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));
这里我贴出了全部的函数代码,以方便你了解产生了甚么,使用lambda可以大大简化代码长度:
query("Hello, world!") .flatMap(urls -> Observable.from(urls)) .subscribe(url -> System.out.println(url));
flatMap()是否是看起来很奇怪?为何它要返回另外1个Observable呢?理解flatMap的关键点在于,flatMap输出的新的Observable正是我们在Subscriber想要接收的。现在Subscriber不再收到List<String>,而是收到1些列单个的字符串,就像Observable.from()的输出1样。

这部份也是我当初学RxJava的时候最难理解的部份,1旦我突然领悟了,RxJava的很多疑问也就1并解决了。

还可以更好

flatMap()实在不能更赞了,它可以返回任何它想返回的Observable对象。
比以下面的方法:
// 返回网站的标题,如果404了就返回null Observable<String> getTitle(String URL);
接着前面的例子,现在我不想打印URL了,而是要打印收到的每一个网站的标题。问题来了,我的方法每次只能传入1个URL,并且返回值不是1个String,而是1个输出String的Observabl对象。使用flatMap()可以简单的解决这个问题。
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));
使用lambda:
query("Hello, world!") .flatMap(urls -> Observable.from(urls)) .flatMap(url -> getTitle(url)) .subscribe(title -> System.out.println(title));
是否是感觉很不可思议?我居然能将多个独立的返回Observable对象的方法组合在1起!帅呆了!
不止这些,我还将两个API的调用组合到1个链式调用中了。我们可以将任意多个API调用链接起来。大家应当都应当知道同步所有的API调用,然后将所有API调用的回调结果组合成需要展现的数据是1件多么蛋疼的事情。这里我们成功的避免了callback hell(多层嵌套的回调,致使代码难以浏览保护)。现在所有的逻辑都包装成了这类简单的响应式调用。

丰富的操作符

目前为止,我们已接触了两个操作符,RxJava中还有更多的操作符,那末我们如何使用其他的操作符来改进我们的代码呢?
getTitle()返回null如果url不存在。我们不想输出"null",那末我们可以从返回的title列表中过滤掉null值!
query("Hello, world!") .flatMap(urls -> Observable.from(urls)) .flatMap(url -> getTitle(url)) .filter(title -> title != null) .subscribe(title -> System.out.println(title));
filter()输出和输入相同的元素,并且会过滤掉那些不满足检查条件的。

如果我们只想要最多5个结果:
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()输出最多指定数量的结果。

如果我们想在打印之前,把每一个标题保存到磁盘:
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()允许我们在每次输出1个元素之前做1些额外的事情,比如这里的保存标题。

看到这里操作数据流是多么简单了么。你可以添加任意多的操作,并且不会弄乱你的代码。

RxJava包括了大量的操作符。操作符的数量是有点吓人,但是很值得你去挨个看1下,这样你可以知道有哪些操作符可使用。弄懂这些操作符可能会花1些时间,但是1旦弄懂了,你就完全掌握了RxJava的威力。

你乃至可以编写自定义的操作符!这篇blog不打算将自定义操作符,如果你想的话,清自行Google吧。

感觉如何?

好吧,你是1个怀疑主义者,并且还很难被说服,那为何你要关心这些操作符呢?

由于操作符可让你对数据流做任何操作。

将1系列的操作符链接起来就能够完成复杂的逻辑。代码被分解成1系列可以组合的片断。这就是响应式函数编程的魅力。用的越多,就会越多的改变你的编程思惟。


另外,RxJava也使我们处理数据的方式变得更简单。在最后1个例子里,我们调用了两个API,对API返回的数据进行了处理,然后保存到磁盘。但是我们的Subscriber其实不知道这些,它只是认为自己在接收1个Observable<String>对象。良好的封装性也带来了编码的便利!


在第3部份中,我将介绍RxJava的另外1些很酷的特性,比如毛病处理和并发,这些特性其实不会直接用来处理数据。

原文链接
























------分隔线----------------------------
------分隔线----------------------------

最新技术推荐