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上来,所以这点要坚信!!
RxJava——响应式编程的更多相关文章
- RxJava响应式编程,入门的HelloWorld;
RxJava核心就是异步,它也被称之为响应式编程:最大的优势就是随着程序逻辑变得越来越复杂,它依然能够保持简洁. Rxjava真的是让人又爱又恨,因为它的线程切换和链式调用真的很好用,但是入门却有点难 ...
- RxJava(一):响应式编程与Rx
一,响应式编程 响应式编程是一种关注于数据流(data streams)和变化传递(propagation of change)的异步编程方式. 1.1 异步编程 传统的编程方式是顺序执行的,必须在完 ...
- 响应式编程库RxJava初探
引子 在读 Hystrix 源码时,发现一些奇特的写法.稍作搜索,知道使用了最新流行的响应式编程库RxJava.那么响应式编程究竟是怎样的呢? 本文对响应式编程及 RxJava 库作一个初步的探索. ...
- iOS开发--Swift RAC响应式编程初探
时间不是很充足, 先少说点, RAC的好处是响应式编程, 不需要自己去设置代理委托, target, 而是主要以信息流(signal), block为主, 看到这里激动吧, 它可以帮你监听你的事件, ...
- iOS响应式编程:ReactiveCocoa vs RxSwift 选谁好
转载: iOS响应式编程:ReactiveCocoa vs RxSwift 选谁好 内容来自stack overflow的一个回答:ReactiveCocoa vs RxSwift – pros an ...
- (转)Spring Boot 2 (十):Spring Boot 中的响应式编程和 WebFlux 入门
http://www.ityouknow.com/springboot/2019/02/12/spring-boot-webflux.html Spring 5.0 中发布了重量级组件 Webflux ...
- Spring Boot 2 (十):Spring Boot 中的响应式编程和 WebFlux 入门
Spring 5.0 中发布了重量级组件 Webflux,拉起了响应式编程的规模使用序幕. WebFlux 使用的场景是异步非阻塞的,使用 Webflux 作为系统解决方案,在大多数场景下可以提高系统 ...
- 【SpringBoot】SpringBoot2.0响应式编程
========================15.高级篇幅之SpringBoot2.0响应式编程 ================================ 1.SprinBoot2.x响应 ...
- WebFlux基础之响应式编程
上篇文章,我们简单的了解了WebFlux的一些基础与背景,并通过示例来写了一个demo.我们知道WebFlux是响应式的web框架,其特点之一就是可以通过函数式编程方式配置route.另外究竟什么是响 ...
随机推荐
- Go语言中使用切片(slice)实现一个Vector容器
Go语言中的切片(slice)和一些内置函数能实现其他语言容器类Array.Vector的功能,但是Go内置语言包container里只提供了list.heap.ring三种容器,缺少vector容器 ...
- 【VS开发】#pragma预处理命令
#pragma预处理命令 #pragma可以说是C++中最复杂的预处理指令了,下面是最常用的几个#pragma指令: #pragma comment(lib,"XXX.lib") ...
- 通过js获取本机的IP地址
参考链接:https://blog.csdn.net/qq_39327418/article/details/90052668
- 拯救你丢失的精度——BigInteger和BigDecimal类(入门)
第三阶段 JAVA常见对象的学习 BigInteger和BigDecimal类 BigInteger类 (一) 构造方法: //针对超过整数范围的运算(整数最大值:2147483647) BigInt ...
- kafka producer 生产者客户端参数配置
在生产者向broker发送消息时,需要配置不同的参数来确保发送成功. acks = all #指定分区中有多少副本必须收到这条消息,生产者才认为这条消息发送成功 acks = 0 #生产者发送消息之后 ...
- 【转贴】NUMA的取舍与优化设置
NUMA的取舍与优化设置 https://www.cnblogs.com/tcicy/p/10191505.html 在os层numa关闭时,打开bios层的numa会影响性能,QPS会下降15-30 ...
- 【AI】【人工智能】【计算机】人工智能工程技术人员等职业信息公示
人社厅发[2019]48号 各省.自治区.直辖市及新疆生产建设兵团人力资源社会保障厅(局).市场监管局.统计局,国务院各部门.各直属机构.各中央企业.有关社会组织人事劳动保障工作机构,中央军委政治工作 ...
- Graphite简要教程
转载自DevOps实战:Graphite监控上手指南 英文原文Getting Started with Monitoring using Graphite 英文原文Google快照 作者 Frankl ...
- 怎样设置HTTP请求头Header
使用: xhr.setRequestHeader(); xhr.setRequestHeader('Content-Type', 'application/json'); xhr.setRequest ...
- [JZOJ3521]道路覆盖--状压DP
题目链接 略略略 分析 首先一看到使得最低的高度最高就想到了二分,于是就转化成了一个是否可行的问题 发现这个\(k\)都很小,考虑使用状态压缩DP 但是我一开始发现似乎并不好设计状态...如果这个\( ...