初识Android的ReactiveX
初识Android的ReactiveX
开发一个复杂一点的Android应用都会用到网络请求,交互和动画。这些都意味着
要写很多的回调嵌套。这样的代码也被称为callback hell(回调地狱)。这样的
代码不仅长,很难理解,而且也是错误高发的地方。ReactiveX
提供了一个清晰、准确处理异步问题和事件的方法。
RxJava是一个ReactiveX在JVM上的实现,由NetFlix开发。这个库在Java开发者中
广为流传。这个教程中你会学到如何在Android应用开发中使用RxJava。这里Android中的RxJava
可以简称为RxAndroid。
1. 配置RxAndroid
要在android中使用RxAndroid,需要在build.gradle里添加compile依赖项。
compile 'io.reactivex:rxjava:1.1.1'
2. Oberser和Observable的基础
在使用RxJava的时候,你会经常遇到Observer
和Observable
的概念。
你可以把Observable
理解为数据的生产者,而Observer
则可以理解为数据的消费者。在RxJava里
数据的生产者都是Observable
类的实例,消费者都是接口Observer
的实例。
类Observable
有很多静态方法,称为operator。来创建类Observable
的实例。以下
代码演示了如何使用方法just
方法(这就是上文说到的operator)来穿件一个非常简单
的实例,并提供一个String
数据:
Observable<String> theObservable = Observable.just("hello world!");
我们刚刚创建的实例会在用有至少一个观察者的时候发出数据。要创建一个观察者,就需要创建
一个实现了接口Observer
的类。接口Observer
里的方法可以处理从Observable
实例发出的通知。
下面的观察者可以把Observable
实例发出的数据打印出来。
private final static String TAG = RxJavaActivity.class.getSimpleName();
Observer<String> theObserver = new Observer<String>() {
@Override
public void onCompleted() {
Log.d(TAG, "completed");
}
@Override
public void onError(Throwable e) {
Log.d(TAG, "error");
}
@Override
public void onNext(String s) {
Log.d(TAG, "data:- " + s);
}
};
- 在Observable实例没有数据在发出的时候调用。
- 在Observable实例遇到错误的时候调用。
- 在Observable实例每次发出数据的时候调用。
要给观察者指定一个observable的实例,我们需要调用subscribe
方法。这个方法会返回一个Subscription
实例。下面的代码让theObserver
观察者观察theObservable
。
Subscription theSubscription = theObservable.subscribe(theObserver);
当观察者添加到了observable实例中,observable实例就发出数据。因此,如果运行上面的代码你会
看到hello world!被打印出来。具体如下:
okhttp.demo.com.okhttpdemo D/RxJavaActivity: data:- hello world!
okhttp.demo.com.okhttpdemo D/RxJavaActivity: completed
一般来说方法onCompleted
和onError
不会用到,不过某些特殊情况也会用到,不过这里不多叙述。
所以,有更加简洁的可以用到Action1
接口,这个接口只有一个方法。
Action1<String> theAction = new Action1<String>() {
@Override
public void call(String s) {
Log.d(TAG, "action1 data:- " + s);
}
};
Subscription theSubscription = theObservable.subscribe(theAction);
这样提交了一个接口action1
的实例之后,Observable
的实例就发出数据。
要从observable中去除一个观察者只需要在Subscription
实例上调用方法unsubscribe
。
theSubscription.unsubscribe();
3. 使用Operator
现在你已经知道如何创建观察者和observable(可观察对象)。下面来看看如何使用RxJava的operator(操作符)
来创建、转换observable的,当然还有其他的一些操作。现在创建一个复杂一点的Observable
对象。
这个对象发出一个Integer
数组数据。方法from
可以这个功能。
Observable<Integer> listObservable = Observable.from(new Integer[]{1, 2, 3, 4, 5});
listObservable.subscribe(new Action1<Integer>() {
@Override
public void call(Integer integer) {
Log.d(TAG, "data:- " + String.valueOf(integer));
}
});
运行这段代码你会发现数组中的数据一个接一个的打印了出来。
okhttp.demo.com.okhttpdemo D/RxJavaActivity: data:- 1
okhttp.demo.com.okhttpdemo D/RxJavaActivity: data:- 2
okhttp.demo.com.okhttpdemo D/RxJavaActivity: data:- 3
okhttp.demo.com.okhttpdemo D/RxJavaActivity: data:- 4
okhttp.demo.com.okhttpdemo D/RxJavaActivity: data:- 5
如果你熟悉javascript或者kotlin, 你就会对map
和filter
方法如何在数组中使用有一定的认识。
RxJava也有类似的operator来处理observable(可观察对象)。由于java 7没有lambda表达式,我们需要
模拟一下这个lambda表达式。要模拟只接受一个参数的lambda表达式只要创建一个实现了接口Func1
的实例。
这里你可以用map
方法来遍历listObservable
的每一个元素。这个例子会遍历observable每个整数并输出这个数字的平方值。
listObservable.map(new Func1<Integer, Integer>() { // 1
@Override
public Integer call(Integer integer) {
return integer * integer; // 2
}
});
- 输入和输出的值都是
Integer
类型的。 - 输出数字的平方值。
这里有一点需要注意的地方。map
方法返回的是一个新的Observable
对象,这个方法本身并不改变
原本的Observable
的实例。如果给listObsevable
添加了观察者,就会返回这些数字的平方值。
Operator(操作符)可以成链式调用。比如,以下代码使用了skip
方法跳过前两个数字,并使用filter
方法
来忽略奇数:
// skip & filter
listObservable
.skip(2) // 1
.filter(new Func1<Integer, Boolean>(){
@Override
public Boolean call(Integer integer) {
return integer % 2 == 0; // 2
}
});
输出:
okhttp.demo.com.okhttpdemo D/RxJavaActivity: skip & filter data:- 4
4. 该处理异步问题了
我们前面创建的observer(观察者)和observable(可观察对象)都是在单个的线程中运行的,都在Android的UI线程中。
这里,我们要用RxJava来处理一些多线程的问题。同时,演示解决callback hell(回调地狱)的问题。
假设我们有一个叫做fetchData
的方法,这个方法被用来从API上获取数据。这个方法接受一个URL字符串
为参数,并返回一个字符串。这个方法可以这么使用:
String content = fetchData("https://api.github.com/orgs/octokit/repos");
如果你执行命令curl -i https://api.github.com/orgs/octokit/repos是会返回一个json字符串的。
是的,这个URL是github的API的一个例子。在这里用正合适。
在Android里,网络请求不能在UI线程中,只能另外开辟一个线程或者使用AsyncTask
。有了RxJava之后
你就多了一个选项。是用subscribeOn
和observeOn
操作符(operator),你可以显示的指出后台任务在哪个线程
运行,更新界面的任务在哪个线程(当然这必须是在UI线程)。
不过在继续往下之前,我们需要把RxAndroid的扩展添加到项目依赖里。这个库是RxJava在Android上的扩展。
compile 'io.reactivex:rxandroid:1.1.0'
下面的代码演示如何使用create
操作符(operator)创建一个Observable
对象。使用这个方法创建的
Observable
对象必须实现Observable.OnSubscribe
接口,并调用onNext
, onError
和onComplete
来控制发出什么数据等操作。
Observable<String> asyncObservable = Observable.create(new Observable.OnSubscribe<String>() {
@Override
public void call(Subscriber<? super String> subscriber) {
try {
String content = fetchData("https://api.github.com/orgs/octokit/repos");
subscriber.onStart();
// 1
subscriber.onNext(content);
// 2
subscriber.onCompleted();
} catch (Exception e) {
// 3
subscriber.onError(e);
}
}
});
- 把方法
fetchData
获取的数据发出去。 - 这里没有什么要做的。
- 如果出现了错误,那么就发出错误信息。
然后就可以给这个asyncObservable
, Observable
对象来调用subscribeOn
和observeOn
两个方法来指定执行的
线程。
Observable<String> asyncObservable = Observable.create(new Observable.OnSubscribe<String>() {
@Override
public void call(Subscriber<? super String> subscriber) {
try {
String content = fetchData("https://api.github.com/orgs/octokit/repos");
subscriber.onStart();
subscriber.onNext(content);
subscriber.onCompleted();
} catch (Exception e) {
subscriber.onError(e);
}
}
});
asyncObservable
.subscribeOn(Schedulers.newThread())
.observeOn(AndroidSchedulers.mainThread())
.subscribe(new Action1<String>() {
@Override
public void call(String s) {
Log.d(TAG, "async data:- " + s);
Toast.makeText(RxJavaActivity.this, "async data:- " + s, Toast.LENGTH_SHORT).show();
}
});
你也许觉得这个没见的比AsyncTask
或者Thread
+Handler
的组合好。是的,如果你只是需要简单的需要一个
后台运行的线程,不用RxJava也可以。
那么我们考虑一个复杂一点的场景,你需要从两个或以上不同的API并行的获取数据,并且只有在这些API的数据全部都返回回来之后才
才更新一个view。如果你使用传统的方式来处理,你需要些很多不必要的代码以确保这些请求都完成而且没有什么错误。
再考虑另外一个场景,一个后台任务需要在前一个后台任务成功执行之后开始执行。如果用传统的方法,这
将会走进回调地狱(callback hell)。
使用RxJava的operator(操作符),两个场景都可以优雅的解决。比如,如果用fetchData
方法从两个不同的站点获取数据,假设这两个站点是
Yahoo和Google。这就需要创建两个Observable
对象,并使用subscribeOn
方法让他们裕兴在不同的线程上。
yahooObservable.subscribeOn(Schedulers.newThread());
googleObservable.subscribeOn(Schedulers.newThread());
要处理第一个场景,两个请求需要并行的运行。我们可以使用zip
操作符(operator)然后添加观察者。
yahooObservable.subscribeOn(Schedulers.newThread());
googleObservable.subscribeOn(Schedulers.newThread());
Observable<String> zippedObservable = Observable.zip(yahooObservable, googleObservable, new Func2<String, String, String>() {
@Override
public String call(String yahooString, String appleString) {
String result = yahooString + "\n" + appleString;
Log.d(TAG, result);
return result;
}
});
要处理后面一种场景的问题,可以使用方法concat
操作符(operator)来运行有依赖关系的两个线程。
Observable.concat(yahooObservable, googleObservable)
.subscribe(new Action1<String>() {
@Override
public void call(String s) {
Log.d(TAG, "concat data: " + s);
Toast.makeText(RxJavaActivity.this, "concat data: " + s, Toast.LENGTH_SHORT).show();
}
});
5. 处理事件
RxAndroid有一个类叫做ViewObservable
。专门用来方便处理view对象的各种相关事件。下面的
代码将演示如何使用ViewObservable
对象来处理Button
的点击事件。
由于RxAndroid的改动,你需要给项目的依赖项里添加新的库:
compile 'com.jakewharton.rxbinding:rxbinding:0.4.0'
Button eventButton = (Button) findViewById(R.id.event_button);
RxView.clicks(eventButton).subscribe(new Action1<Void>() {
@Override
public void call(Void aVoid) {
Toast.makeText(RxJavaActivity.this, "Clicked", Toast.LENGTH_SHORT).show();
}
});
RxView.clicks(eventButton)
返回一个Observable
的对象。我们可以给这个对象调用各种前文
学到的操作符(operator)。比如,我们要按钮跳过前三次点击,可以这么做:
RxView.clicks(eventButton)
.skip(3)
.subscribe(new Action1<Void>() {
@Override
public void call(Void aVoid) {
Toast.makeText(RxJavaActivity.this, "Clicked", Toast.LENGTH_SHORT).show();
}
});
最后
这里用到了RxJava的观察者Observer
、Observable
还有相关的操作符(operator)来处理异步操作和
事件。使用Rxjava会涉及到函数式编程、响应式编程等一些Android开发者几乎不会涉及到的编程模式。
第一次接触难免会遇到一些困难,这都无所谓。只要继续学习相关内容,都会习以为常。
原文中的内容很多已经不再可用。不过这里已经补齐了各种库修改之后的依赖项并按照修改之后的API重写了全部相关的示例代码。
相关代码都在这里。
初识Android的ReactiveX的更多相关文章
- 初识 Android
创建博客有一年的时间了,一直没把它用起来,颇感惭愧.近日突感有写博客的冲动,更可怕的是这种冲动似乎比我体内的洪荒之力更为凶猛.于是乎,这篇博客悄然诞生.废话不多说,进入正题--初识Android. 这 ...
- 初识Android Studio
刚开始接触Android Studio,很多不适应的地方,自己慢慢摸索,记录下了一些问题和解决途径. 为了能使用android虚拟机,需要下载镜像,镜像有基于arm架构的也有基于intelx86.x6 ...
- 初识android中的动画
动画效果可以大大提高界面的交互效果,因此,动画在移动开发中的应用场景较为普遍.掌握基本的动画效果在成熟的软件开发中不可或缺.除此之外,用户对于动画的接受程度远高于文字和图片,利用动画效果可以加深用户对 ...
- 初识Android NDK
本文介绍Windows环境下搭建Android NDK开发环境,并创建一个简单的使用Native代码的Android Application. 一.环境搭建 二.JNI函数绑定 三.例子 一.环境搭建 ...
- 初识Android && 搭建Android开发环境
搭建Android开发环境需要的工具: JDK(Java Development Kit) Eclipse Android Sdk(Software Development Kit) ADT (And ...
- 1.2……初识Android开发
Android体系结构 Dalvik VM(Android下的java虚拟机)与传统的JVM的区别 传统JVM 基于堆栈的架构 编写.java文件--->编译为.class文件--->打包 ...
- (一)初识Android
第一节:手机操作系统简介 目前的主流智能操作系统有:Android , IOS , windows mobile ; Android 开源,属于谷歌公司,市场份额较大,前景广阔: IOS 属于苹果公司 ...
- 初识Android
#Android项目的目录结构 Activity:应用被打开时显示的界面 src:项目代码 R.java:项目中所有资源文件的资源id Android.jar:Android的jar包,导入此包方可使 ...
- 初识-Android之智能短信项目相关技术整理
标签页切换采用传统的TabHost: 采用TabActivty实现TabHost. 效果图-后补: 相关技术详解推荐: http://blog.csdn.net/zhouli_05/article/d ...
随机推荐
- maven项目运行报错invalid LOC header (bad signature)
切换到项目目录pom.xml文件夹 执行以下命令: mvn test -e -X 找到出错信息 进入目录删除文件. 在ide里面重新部署项目即可.
- node.js下载安装
1.下载node.js在node中文网站,官方网站下载太慢 2.接着让我们点击下载链接,页面上呈现出你所需要下载的安装包,我们这里选择windows x64的安装包进行下载 3.安装node.js,一 ...
- DevExpress XPO 开发指南 简要
最近在看devexpress 安装程序中的代码Demos .. C:\Users\Public\Documents\DevExpress Demos 16.1\Components\WinFor ...
- C# oracle 日期型字段,使用参数传值时,遇到ORA-01810: format code appears twice错误
C#操作oracle数据库时,发现使用to_date('2014-01-03 18:00:00','yyyy-mm-dd hh:MM:ss')时,会出现ORA-01810: format code a ...
- Business.Startup.Learning from Startup Mistakes at SpringSource
http://www.infoq.com/news/2014/07/startup-spring
- c++文件的输入输出
emmm,错误地方还请指出(以下代码复制粘贴会报错,我用codeblocks测试过,不知道为什么qaq) 头文件#include < fstream > 这里ofstream是" ...
- HDU 2255.奔小康赚大钱 最大权匹配
奔小康赚大钱 Time Limit: 1000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total Subm ...
- netstat 查看端口命令
查看特定端口是否启动 netstat -lnp |
- 2018年设计师都在用的PS切图插件--摹客iDoc
终于找到你,我梦寐以求的PS切图插件.曾几何时,设计师在完成设计稿之后高效的输出标注切图一直是设计师的噩梦.为什么这么说呢?开发要的那么多尺寸,我到底该怎么切图?iPhone的版本已经不少了,更别提安 ...
- Flex + .Net从本地选择一个图片上传到服务器
<mx:TextInput id="TxtFileName" editable="false" width="200"/> &l ...