初识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的时候,你会经常遇到ObserverObservable的概念。
你可以把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);
    }
};
  1. 在Observable实例没有数据在发出的时候调用。
  2. 在Observable实例遇到错误的时候调用。
  3. 在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

一般来说方法onCompletedonError不会用到,不过某些特殊情况也会用到,不过这里不多叙述。
所以,有更加简洁的可以用到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, 你就会对mapfilter方法如何在数组中使用有一定的认识。
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
    }
});
  1. 输入和输出的值都是Integer类型的。
  2. 输出数字的平方值。

这里有一点需要注意的地方。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之后
你就多了一个选项。是用subscribeOnobserveOn操作符(operator),你可以显示的指出后台任务在哪个线程
运行,更新界面的任务在哪个线程(当然这必须是在UI线程)。

不过在继续往下之前,我们需要把RxAndroid的扩展添加到项目依赖里。这个库是RxJava在Android上的扩展。

compile 'io.reactivex:rxandroid:1.1.0'

下面的代码演示如何使用create操作符(operator)创建一个Observable对象。使用这个方法创建的
Observable对象必须实现Observable.OnSubscribe接口,并调用onNextonErroronComplete
来控制发出什么数据等操作。

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);
        }
    }
});
  1. 把方法fetchData获取的数据发出去。
  2. 这里没有什么要做的。
  3. 如果出现了错误,那么就发出错误信息。

然后就可以给这个asyncObservableObservable对象来调用subscribeOnobserveOn两个方法来指定执行的
线程。

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的观察者ObserverObservable还有相关的操作符(operator)来处理异步操作和
事件。使用Rxjava会涉及到函数式编程、响应式编程等一些Android开发者几乎不会涉及到的编程模式。
第一次接触难免会遇到一些困难,这都无所谓。只要继续学习相关内容,都会习以为常。

原文中的内容很多已经不再可用。不过这里已经补齐了各种库修改之后的依赖项并按照修改之后的API重写了全部相关的示例代码。

相关代码都在这里

欢迎加群互相学习,共同进步。QQ群:iOS: 58099570 | Android: 330987132 | Go:217696290 | Python:336880185 | 做人要厚道,转载请注明出处!http://www.cnblogs.com/sunshine-anycall/p/5213552.html
相关文章
|
21天前
|
Java Android开发
Android 开发获取通知栏权限时会出现两个应用图标
Android 开发获取通知栏权限时会出现两个应用图标
12 0
|
1月前
|
XML 缓存 Android开发
Android开发,使用kotlin学习多媒体功能(详细)
Android开发,使用kotlin学习多媒体功能(详细)
97 0
|
2月前
|
Android开发
安卓SO层开发 -- 编译指定平台的SO文件
安卓SO层开发 -- 编译指定平台的SO文件
30 0
|
1月前
|
设计模式 人工智能 开发工具
安卓应用开发:构建未来移动体验
【2月更文挑战第17天】 随着智能手机的普及和移动互联网技术的不断进步,安卓应用开发已成为一个热门领域。本文将深入探讨安卓平台的应用开发流程、关键技术以及未来发展趋势。通过分析安卓系统的架构、开发工具和框架,本文旨在为开发者提供全面的技术指导,帮助他们构建高效、创新的移动应用,以满足不断变化的市场需求。
18 1
|
1月前
|
机器学习/深度学习 调度 Android开发
安卓应用开发:打造高效通知管理系统
【2月更文挑战第14天】 在移动操作系统中,通知管理是影响用户体验的关键因素之一。本文将探讨如何在安卓平台上构建一个高效的通知管理系统,包括服务、频道和通知的优化策略。我们将讨论最新的安卓开发工具和技术,以及如何通过这些工具提高通知的可见性和用户互动性,同时确保不会对用户造成干扰。
33 1
|
12天前
|
XML 开发工具 Android开发
构建高效的安卓应用:使用Jetpack Compose优化UI开发
【4月更文挑战第7天】 随着Android开发不断进化,开发者面临着提高应用性能与简化UI构建流程的双重挑战。本文将探讨如何使用Jetpack Compose这一现代UI工具包来优化安卓应用的开发流程,并提升用户界面的流畅性与一致性。通过介绍Jetpack Compose的核心概念、与传统方法的区别以及实际集成步骤,我们旨在提供一种高效且可靠的解决方案,以帮助开发者构建响应迅速且用户体验优良的安卓应用。
|
15天前
|
Java Android开发
Android开发之使用OpenGL实现翻书动画
本文讲述了如何使用OpenGL实现更平滑、逼真的电子书翻页动画,以解决传统贝塞尔曲线方法存在的卡顿和阴影问题。作者分享了一个改造后的外国代码示例,提供了从前往后和从后往前的翻页效果动图。文章附带了`GlTurnActivity`的Java代码片段,展示如何加载和显示书籍图片。完整工程代码可在作者的GitHub找到:https://github.com/aqi00/note/tree/master/ExmOpenGL。
19 1
Android开发之使用OpenGL实现翻书动画
|
15天前
|
Android开发 开发者
Android开发之OpenGL的画笔工具GL10
这篇文章简述了OpenGL通过GL10进行三维图形绘制,强调颜色取值范围为0.0到1.0,背景和画笔颜色设置方法;介绍了三维坐标系及与之相关的旋转、平移和缩放操作;最后探讨了坐标矩阵变换,包括设置绘图区域、调整镜头参数和改变观测方位。示例代码展示了如何使用这些方法创建简单的三维立方体。
12 1
Android开发之OpenGL的画笔工具GL10
|
21天前
|
Android开发
Android开发小技巧:怎样在 textview 前面加上一个小图标。
Android开发小技巧:怎样在 textview 前面加上一个小图标。
12 0
|
21天前
|
Android开发
Android 开发 pickerview 自定义选择器
Android 开发 pickerview 自定义选择器
12 0