初识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
相关文章
|
1月前
|
缓存 搜索推荐 Android开发
安卓开发中的自定义控件实践
【10月更文挑战第4天】在安卓开发的海洋中,自定义控件是那片璀璨的星辰。它不仅让应用界面设计变得丰富多彩,还提升了用户体验。本文将带你探索自定义控件的核心概念、实现过程以及优化技巧,让你的应用在众多竞争者中脱颖而出。
|
1天前
|
XML 存储 Java
探索安卓开发之旅:从新手到专家
【10月更文挑战第35天】在数字化时代,安卓应用的开发成为了一个热门话题。本文旨在通过浅显易懂的语言,带领初学者了解安卓开发的基础知识,同时为有一定经验的开发者提供进阶技巧。我们将一起探讨如何从零开始构建第一个安卓应用,并逐步深入到性能优化和高级功能的实现。无论你是编程新手还是希望提升技能的开发者,这篇文章都将为你提供有价值的指导和灵感。
|
1天前
|
Android开发
布谷语音软件开发:android端语音软件搭建开发教程
语音软件搭建android端语音软件开发教程!
|
9天前
|
编解码 Java Android开发
通义灵码:在安卓开发中提升工作效率的真实应用案例
本文介绍了通义灵码在安卓开发中的应用。作为一名97年的聋人开发者,我在2024年Google Gemma竞赛中获得了冠军,拿下了很多项目竞赛奖励,通义灵码成为我的得力助手。文章详细展示了如何安装通义灵码插件,并通过多个实例说明其在适配国际语言、多种分辨率、业务逻辑开发和编程语言转换等方面的应用,显著提高了开发效率和准确性。
|
8天前
|
Android开发 开发者 UED
安卓开发中自定义View的实现与性能优化
【10月更文挑战第28天】在安卓开发领域,自定义View是提升应用界面独特性和用户体验的重要手段。本文将深入探讨如何高效地创建和管理自定义View,以及如何通过代码和性能调优来确保流畅的交互体验。我们将一起学习自定义View的生命周期、绘图基础和事件处理,进而探索内存和布局优化技巧,最终实现既美观又高效的安卓界面。
20 5
|
6天前
|
JSON Java Android开发
探索安卓开发之旅:打造你的第一个天气应用
【10月更文挑战第30天】在这个数字时代,掌握移动应用开发技能无疑是进入IT行业的敲门砖。本文将引导你开启安卓开发的奇妙之旅,通过构建一个简易的天气应用来实践你的编程技能。无论你是初学者还是有一定经验的开发者,这篇文章都将成为你宝贵的学习资源。我们将一步步地深入到安卓开发的世界中,从搭建开发环境到实现核心功能,每个环节都充满了发现和创造的乐趣。让我们开始吧,一起在代码的海洋中航行!
|
7天前
|
缓存 数据库 Android开发
安卓开发中的性能优化技巧
【10月更文挑战第29天】在移动应用的海洋中,性能是船只能否破浪前行的关键。本文将深入探讨安卓开发中的性能优化策略,从代码层面到系统层面,揭示如何让应用运行得更快、更流畅。我们将以实际案例和最佳实践为灯塔,引领开发者避开性能瓶颈的暗礁。
22 3
|
10天前
|
存储 IDE 开发工具
探索Android开发之旅:从新手到专家
【10月更文挑战第26天】在这篇文章中,我们将一起踏上一段激动人心的旅程,探索如何在Android平台上从零开始,最终成为一名熟练的开发者。通过简单易懂的语言和实际代码示例,本文将引导你了解Android开发的基础知识、关键概念以及如何实现一个基本的应用程序。无论你是编程新手还是希望扩展你的技术栈,这篇文章都将为你提供价值和启发。让我们开始吧!
|
15天前
|
Java API Android开发
安卓应用程序开发的新手指南:从零开始构建你的第一个应用
【10月更文挑战第20天】在这个数字技术不断进步的时代,掌握移动应用开发技能无疑打开了一扇通往创新世界的大门。对于初学者来说,了解并学习如何从无到有构建一个安卓应用是至关重要的第一步。本文将为你提供一份详尽的入门指南,帮助你理解安卓开发的基础知识,并通过实际示例引导你完成第一个简单的应用项目。无论你是编程新手还是希望扩展你的技能集,这份指南都将是你宝贵的资源。
44 5
|
14天前
|
设计模式 IDE Java
探索安卓开发:从新手到专家的旅程
【10月更文挑战第22天】 在数字时代的浪潮中,移动应用开发如同一座金矿,吸引着无数探险者。本文将作为你的指南针,指引你进入安卓开发的广阔天地。我们将一起揭开安卓平台的神秘面纱,从搭建开发环境到掌握核心概念,再到深入理解安卓架构。无论你是初涉编程的新手,还是渴望进阶的开发者,这段旅程都将为你带来宝贵的知识和经验的财富。让我们开始吧!
下一篇
无影云桌面