RxJava 简单实战

都说RxJava 是非常强大但是难于上手的。我接触RxJava已经有一段时间了,今天就从自己的项目中,将用到RXJava的部分单独的拿出来写一篇文章,用来帮助看了很多RxJava相关的文章但是还不知道怎么去使用的同学。

前言

阅读本文章之前,我们在回顾或者加强几个基本概念。
Observer:观察者
Observable:可观察者
Subscribe:订阅
observalbe(观察者) subscribe(订阅) observer(被观察者)

Tips
上面的逻辑看起来和我们正常的逻辑是相反的,按照常理来说不应该是被观察者订阅观察者吗?为什么反过来了,具体原因可以在 给Android开发者的 RxJava 详解 中找到答案

RxJava使用三步走

RxJava基本实现只需要三步

  • 创建Observer
  • 创建Observable
  • 订阅

1.创建Observer

Observer即观察者,他决定事件触发的时候将会有什么样的行为。基本的Observer我们可以这么实现:

Observer<String> observer = new Observer<String>() {
    @Override
    public void onNext(String s) {
        Log.d(tag, "Item: " + s);
    }

    @Override
    public void onCompleted() {
        Log.d(tag, "Completed!");
    }

    @Override
    public void onError(Throwable e) {
        Log.d(tag, "Error!");
    }
};

2.创建Observable

Observable 即被观察者,他决定什么时候触发怎样的事件。
我们可以使用create()方法创建一个Observable

Observable observable = Observable.create(new Observable.OnSubscribe<String>() {
    @Override
    public void call(Subscriber<? super String> subscriber) {
        subscriber.onNext("Hello");
        subscriber.onNext("Hi");
        subscriber.onNext("Aloha");
        subscriber.onCompleted();
    }
});

更简单的,我们可以使用just(T...)创建一个Observable

Observable observable = Observable.just("Hello", "Hi", "Aloha");
// 将会依次调用:
// onNext("Hello");
// onNext("Hi");
// onNext("Aloha");
// onCompleted();

也可以使用from(T[])来创建一个Observable

String[] words = {"Hello", "Hi", "Aloha"};
Observable observable = Observable.from(words);

3.订阅

我们创建了ObservableObserver之后,在用subscribe()将他们链接起来,代码就可以工作啦。

observable.subscribe(observer);

在我自己的项目中RxJava使用场景举例

RxJava与Retrofit结合

这里比较简单,只需要稍微改变Retrofit请求接口方法的返回值类型就好了。

@GET("openapi.do?keyfrom=xxx&key=xxx&type=data&doctype=json&version=1.1")
Observable<YouDaoResult> getTranslationYouDao(@Query("q") String q);

接着使用Retrofit对象,创建接口实例,调用接口方法,即可获取Observable。
我在项目中使用的Dagger2,所以看起来和只使用了RxJava与Retrofit的代码有所不同

@Provides
@Singleton
public static ClientApi provideClientApi() {
    Retrofit retrofit = new Retrofit.Builder()
            .baseUrl(BASE_URL)
            .addCallAdapterFactory(RxJavaCallAdapterFactory.create())
            .addConverterFactory(GsonConverterFactory.create())
            .build();
    return retrofit.create(ClientApi.class);
}
public Observable<YouDaoResult> getTranslation(String query) {
        return getApi().getTranslationYouDao(query);
    }

在我的项目中,从网络获取的实体类型是YouDaoResult,本地数据库存储的实体类型是经过简化的Result,在业务逻辑中我想实现在查询一个单词的时候,如果本地数据库已经存在了单词记录就从本地读取记录,而不从网络获取。然而两个实体类型不同,我又想使用优雅的方法解决它,我能不能获取了YouDaoResult之后,立刻就转换成Result呢?后来我使用了RxJava的map()变换对象流方法。

Tips
在我的项目中,所有的Observable都是放在一起管理的,作为DataLayer(数据层),在业务方法中,想要获取数据首先要在数据层中获取Observable,再使用RxJava的方法去处理它。

@Override
public Observable<Result> getTranslation(String query) {
    return getApi().getTranslationYouDao(query)
            .map(new Func1<YouDaoResult, Result>() {
                @Override
                public Result call(YouDaoResult youDaoResult) {
                    return youDaoResult.getResult();
                }
            });
}

在这里要放大招啦,项目中获取单词的方法是怎么实现的。根据代码注释可以很直观的看出RxJava的优点,异步,简洁,即使逻辑复杂,已然可以保持简洁。在查询单词的业务逻辑中,主要做了下面几件事:

  • 在本地数据库有单词数据时优先从本地数据库查询单词
  • 本地数据库没有单词数据则从网络获取数据
  • 单词在输出前进行缓存,这里又分为两步不过实现方法在数据库层。
  • 异步
public void fetchTranslation(String query) {
    // 分发开始刷新列表事件(Flux架构)
    getDispatcher().dispatch(new Action.Builder().with(TranslateActions.ACTION_TRANSLATION_LOADING).build());

    // 本地数据库数据源
    Observable<Result> cache = getDataLayer().getTranslateService().getLocalTranslation(query);

    // 服务端数据源
    Observable<Result> network = getDataLayer().getTranslateService().getTranslation(query);

    // 没有本地数据在使用网络数据
    Observable<Result> source = Observable
            .concat(cache, network)
            // 依次遍历序列中的数据源, 返回第一个符合条件的数据源
            .first(new Func1<Result, Boolean>() {
                @Override
                public Boolean call(Result result) {
                    return result != null;
                }
            });

    // 重新查询数据则更新history列表,在save方法中有判断,具体见TranslateDB
    source = source.doOnNext(new Action1<Result>() {
        @Override
        public void call(Result result) {
            getDataLayer().getTranslateService().saveToHistory(result);
        }
    });

    source.subscribeOn(Schedulers.io())
            .observeOn(AndroidSchedulers.mainThread())
            .subscribe(new Action1<Result>() {
                @Override
                public void call(Result result) {
                    // Flux架构分发事件
                    getDispatcher().dispatch(new Action.Builder()
                            .with(TranslateActions.ACTION_TRANSLATION_FINISH)
                            .bundle(TranslateActions.KEY_TRANSLATION_ANSWER, result)
                            .build());
                }
            }, new Action1<Throwable>() {
                @Override
                public void call(Throwable throwable) {
                    // Flux架构分发事件
                    Action action = new Action.Builder()
                            .with(TranslateActions.ACTION_TRANSLATION_NET_ERROR)
                            .build();
                    dispatcher.dispatch(action);
                }
            });
}

小结

RxJava并没有那么难,我们不敢将它引入到实际开发环境的最终原因只是我们对RxJava没有那么熟悉。Talk is cheap,赶紧去练习吧。
最后放上我的项目地址: Translate
欢迎围观,欢迎批评,欢迎讨论。

延伸阅读

RxJava Github
RxJava官网

中文学习资料:
给 Android 开发者的 RxJava 详解
过滤序列 | RxJava Essentials CN
lzyzsd/Awesome-RxJava: RxJava resources
RxJava 与 Retrofit 结合的最佳实践

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,539评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,911评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,337评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,723评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,795评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,762评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,742评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,508评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,954评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,247评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,404评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,104评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,736评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,352评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,557评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,371评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,292评论 2 352

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,033评论 25 707
  • 我从去年开始使用 RxJava ,到现在一年多了。今年加入了 Flipboard 后,看到 Flipboard 的...
    Jason_andy阅读 5,464评论 7 62
  • 前言我从去年开始使用 RxJava ,到现在一年多了。今年加入了 Flipboard 后,看到 Flipboard...
    占导zqq阅读 9,162评论 6 151
  • 理解用户与用户组 用户:设置权限不让其他人访问自己的文件 用户组:设置权限不让其他用户组的成员访问本用户组文件 r...
    楠昭阅读 232评论 0 0
  • 导语 拥有财富、名声、势力,拥有整个世界的海贼王 – 哥尔罗杰,他在临刑前的一句话,让人们趋之若鹜奔向大海。“想要...
    crystalcj阅读 516评论 5 3