RxJava——响应式编程
自从06年开始,Rxandroid公司项目中陆续就开始使用它了,而它的基础是由Rxjava演变过来的,如今它也是越来越被广泛使用在商业项目中了,而做为"专业"的自己还是一直对它一知半解,而在实现功能时首先并不会想到用它去架构代码,而是还是想用着传统的方式,并不是不想用它,而是对它的使用不太了解有点抗拒,所以接下来准备系统的学下它,以便在未来如果项目中用了Rx相关的东东不至于太陌生,同时也为了自己将来在做项目时能很好的应用它。
在学习一门新技术之前首先得对它有一个大概的认识,也就是理论这方面肯定是少不了的,下面开始!
什么是Rx?
- ReactiveX是Reactive Extensions的缩写,一般简写为Rx,由微软的架构师Erik Meijer领导的团队开发,在2012年11月开源。
- Rx是一个编程模型,目标是提供一致的编程接口,帮助开发者更方便的处理异步数据,目前Rx支付大部分流行的编程语言,比如Java、C#、PHP等。
- Rx是一函数库,让开发者可以利用可观察序列和LINQ风格查询操作符来编写异步和基于事件的程序。
- 可以这样定义:Rx = Oberservables + LNIQ + Schedules。
- ReactiveX不仅仅是编程接口,更是一种编程思想的突破,Rx还影响了其它的程序库、框架以及编程语言。
Rx模式--观察者模式
- 创建:Rx可以方便的创建事件流和数据流。
- 组合:Rx使用查询式的操作符组合和变换数据流。
- 监听:Rx可以订阅任何可以观察的数据流并执行操作。
Rx优点--简洁
- 函数式风格:Rx可以方便的创建事件流和数据流。
- 简化代码:Rx操作符可以将复杂的逻辑简化为很少的几行代码。
- 异步错误处理机制:传统的try/catch没办法处理异步计算,Rx提供了合适的错误处理机制。
- 轻松使用开发:Rx的Observables和Schedulers让开发者可以避免底层线程同步和各种并发问题。
【注】:有些理论可以参考博文:http://blog.csdn.net/u010046908/article/details/50942247
什么是Rxjava?【主题】
Rxjava是ReactiveX在JVM上的一个实现,也就是说Rxjava是用java语言实现的响应式编程,来创建基于事件的异步程序。
提升开发效率,降低维护成本一直是开发团队永恒不变的宗旨,近一年来国内的技术圈子中越来越多开始提及Rxjava,学习和掌握Rxjava已经很有必要。Rxjava能够帮助我们简化代码逻辑,提升代码可读性,这对于提升开发效率,降低后期维护成本很有帮助。
Rxjava正在Android开发中也变得越来越流行【俗称的RxAndroid,这个在我们公司的项目中就已经大量使用到了】,唯一的问题就是上手不容易,尤其我们都对传统的命令式编程比较熟,但是!!只要弄明白了,会发现使用Rxjava真的是太棒了,所以有必要好好学习下它。
学习Rxjava
响应式编程的主要组成部份是observable、operator和subscriber。一般响应式编程的信息流如下所示:
Observable -> Operator1 -> Operator2 -> Operator3 -> Subscriber
其中Observable是事件的生产者,Subscriber是事件最终的消费者,而其数据的转换由一系列的Operator操作符来执行。
注:因为Subscriber通常是在主线程中执行,因此设计上要求其代码尽可能简单,只对事件做出响应(不对事件或者数据进行修改),而修改事件的工作完全由operator来执行。
有了上面的一大堆理论基础之后,下面用代码来直观的感受一下Rxjava的魅力,编写一个Hello World!,这里采用eclipse中的j2se项目来进行学习,因为还木有用到Android相关的东东,脱离开测试比较方便高效。
在正式写代码之前,首先得在网上下一个Rxjava的jar包,Rxjava如今最新版是2.x,但是要学习还得先从1.x开始,所以这里网上下载Rxjava1.3.0,如下:
下载完之后,新建工程,并将其引入到工程中,接下来就可以开始我们的Hello World啦!!先直观感受下,不必深纠~
public class RxHelloWorld { public static void main(String[] args) { testObservable(); } @SuppressWarnings("deprecation") private static void testObservable() { // 创建被观察者Observable Observable<String> observable = Observable .create(new OnSubscribe<String>() { @Override public void call(Subscriber<? super String> subscriber) { subscriber.onNext("Hello World!"); subscriber.onCompleted(); } }); // 创建观察者Observer Observer<String> observer = new Observer<String>() { @Override public void onCompleted() { System.out.println("observer onCompleted()"); } @Override public void onError(Throwable e) { // TODO Auto-generated method stub } @Override public void onNext(String t) { System.out.println("observer onNext():" + t); } }; // 被观察者订阅(subscribe)观察者 observable.subscribe(observer); } }
编译运行:
可以看出消息是由被观察者Observable发出,由观察者Observer进行消息处理。那如果说发送"Hello World!"时后面要再加一些字符,可以这样修改代码:
其结果肯定如预期:
但是!上面的这个做法不符合Rxjava的思想,这是因为它不希望"在产生数据的地方修改数据",所以可以想到第二种方法,如下:
其结果很显然也是一样的被改变,但是!!!这还是不符合Rxjava的编程思想,那如果要改变一个数据,应该用它的哪种方式呢?答案揭晓---利用Rxjava的操作符来改变数据流,如何做呢?看下面:
@SuppressWarnings("deprecation") private static void testObservable() { // 创建被观察者Observable Observable<String> observable = Observable .create(new OnSubscribe<String>() { @Override public void call(Subscriber<? super String> subscriber) { subscriber.onNext("Hello World!"); subscriber.onCompleted(); } }); // 创建观察者Observer Observer<String> observer = new Observer<String>() { @Override public void onCompleted() { System.out.println("observer onCompleted()"); } @Override public void onError(Throwable e) { // TODO Auto-generated method stub } @Override public void onNext(String t) { System.out.println("observer onNext():" + t); } }; // 被观察者订阅(subscribe)观察者 observable .map(new Func1<String, String>() {//利用map操作符对数据进行修改 @Override public String call(String t) { return t + "cexo"; } }) .subscribe(observer); }
其中的Func1这是个什么东东,好诡异好生涩的样子,这里先只从它的源码有个初步的认识,之后会具体去理解为啥要这样写的:
一个接口声明,先不去过多的理解为啥要这么声明,光从这个代码来看就是做一个类型的转换工作,正好是符合map操作符的意义,了解下既可。
再编译运行:
结果一样,一个这么简单的例子要用多种方式来达到相同的功能,主要是为了说明Rxjava的一种思想,可见跟正常的思路还是不太一样的。
另外这里提到了一个map操作符的概念,可以上官网去瞅一眼:
官网看着还挺漂亮的,看下官网提供的有哪些东东:
可以看到已经有好多语方的版本了,其中第一位就看到了Rxjava啦,接着再看:
这五大东东实际上就是Rxjava的核心,先有个感观上的认识,之会会慢慢去学它们的,那我们关心的操作符就列在第二位:
点开看一下,貌似好多种类,下面大致看一下:
是不是有点晕了,不用太过着急,之后会一点点去学的,有个大概的认识就成。
在官网主页上,看到有这么一个动图:
其中说的是debounce操作符,下面可以点击看一下它的具体介绍:
debounce是过滤操作符中的一种,至于它干嘛用的这里先不用操心,未来会学到的,对于上面这么形象的图有没有想像它是一个动态的图,实际就是动态的:
可见其官网做得是比较好的,另外可以链接到它的Github的官网上来:
打开之后就可以看到有各种相关的项目,其中就可以找到我们的Rxjava啦:
点击进去:
那既然Rxjava1.x最终就要淘汰掉了,那还有必要从Rxjava1.x学起么?很有必要!!!因为2.x就是由1.x演变过来的,核心的东西是没有变的,而且大部份都是1.x的东东,所以把1.x学好了,就可以很平滑快速的过渡到2.x上来,所以这点要坚信!!