RxJava之变换操作符完全解析(四)

话说通过前三篇文章的讲解,想必你对RxJava也有了深刻的认识吧,啥,你说没有?那回去看看!那么今天我们来分析RxJava最牛逼的地方。睁大眼睛看看哦,不管你湿没湿,反正我已经湿了。


Paste_Image.png

一、API

RxJava提供了对事件序列进行变换的支持,这是它的核心功能之一,也是大多数人说RxJava好用的最大原因。所谓变换,就是将事件序列中的对象或整个序列进行加工处理,转换成不同的事件或事件序列。
首先看一个<code>map()</code>的例子:

Observable.just("images/logo.png") // 输入类型 String
    .map(new Func1<String, Bitmap>() {
        @Override
        public Bitmap call(String filePath) { // 参数类型 String
            return getBitmapFromPath(filePath); // 返回类型 Bitmap
        }
    })
    .subscribe(new Action1<Bitmap>() {
        @Override
        public void call(Bitmap bitmap) { // 参数类型 Bitmap
            showBitmap(bitmap);
        }
    });

这里出现了一个叫做 Func1的类。它和 Action1非常相似,也是 RxJava 的一个接口,用于包装含有一个参数的方法。 Func1和Action的区别在于, Func1包装的是有返回值的方法。另外,和 ActionX一样, FuncX也有多个,用于不同参数个数的方法。FuncX和 ActionX的区别在 FuncX包装的是有返回值的方法。
  可以看到,<code>map()</code>方法将参数中的<code>String</code>对象转换成了一个Bitmap对象后返回,经过<code>map()</code>方法后,事件的参数类型也由<code>String</code>转为<code>Bitmap</code>。这种直接变换对象并返回的,就是最常见的变换。不过RxJava的变换远不止这样,它不仅可以针对事件对象,还可以针对整个事件队列,是不是好屌的样子啊?

  • <code>map()</code>:事件对象的直接变换。示意图如下
map示意图
  • <code>flatMap()</code>:FlatMap将一个发射数据的Observable变换为多个Observables,然后将它们发射的数据合并后放进一个单独的Observable.
      这是一个很有用但非常难理解的变换,因此我决定花多些篇幅来介绍它。 首先假设这么一种需求:假设有一个数据结构『学生』,现在需要打印出一组学生的名字。实现方式很简单:
Student[] students = ...;
Subscriber<String> subscriber = new Subscriber<String>() {
    @Override
    public void onNext(String name) {
        Log.d(tag, name);
    }
    ...
};
Observable.from(students)
    .map(new Func1<Student, String>() {
        @Override
        public String call(Student student) {
            return student.getName();
        }
    })
    .subscribe(subscriber);

很简单。那么再假设:如果要打印出每个学生所需要修的所有课程的名称呢?(需求的区别在于,每个学生只有一个名字,但却有多个课程。)首先可以这样实现:

Student[] students = ...;
Subscriber<Student> subscriber = new Subscriber<Student>() {
    @Override
    public void onNext(Student student) {
        List<Course> courses = student.getCourses();
        for (int i = 0; i < courses.size(); i++) {
            Course course = courses.get(i);
            Log.d(tag, course.getName());
        }
    }
    ...
};
Observable.from(students)
    .subscribe(subscriber);

依然很简单。那么如果我不想在 Subscriber中使用 for 循环,而是希望 Subscriber中直接传入单个的 Course对象呢(这对于代码复用很重要)?用 map()显然是不行的,因为 map()是一对一的转化,而我现在的要求是一对多的转化。那怎么才能把一个 Student 转化成多个 Course 呢?这个时候就要用flatMap()了:

Student[] students = ...;
Subscriber<Course> subscriber = new Subscriber<Course>() {
    @Override
    public void onNext(Course course) {
        Log.d(tag, course.getName());
    }
    ...
};
Observable.from(students)
    .flatMap(new Func1<Student, Observable<Course>>() {
        @Override
        public Observable<Course> call(Student student) {
            return Observable.from(student.getCourses());
        }
    })
    .subscribe(subscriber);

从上面看出,<code>flatMap()</code>和<code>map()</code>有一个共同点:它也是把传入的参数转化之后返回另一个对象。但不同的是,<code>flatMap()</code>返回的是<code>Observable</code>对象,并且这个<code>Observable</code>对象并不是直接发给<code>Subscriber</code>的回调方法中。
flatMap()的原理是这样的:
  1. 使用传入的事件对象创建一个 Observable对象;
  2. 并不发送这个 Observable, 而是将它激活,于是它开始发送事件;
  3. 每一个创建出来的 Observable 发送的事件,都被汇入同一个 Observable,而这个 Observable 负责将这些事件统一交给 Subscriber的回调方法。
  这三个步骤,把事件拆成了两级,通过一组新创建 Observable将初始的对象『铺平』之后通过统一路径分发了下去。而这个『铺平』就是 flatMap() 所谓的 flat。
<code>flatMap()</code>示意图:


FlatMap示意图

二、变换的原理:lift()

这些变换虽然功能各有不同,但实质上都是针对事件序列的处理和再发送。而在 RxJava 的内部,它们是基于同一个基础的变换方法:<code>lift(Operator)</code>。首先看一下 <code>lift()</code> 的内部实现(仅核心代码)

// 注意:这不是 lift() 的源码,而是将源码中与性能、兼容性、扩展性有关的代码剔除后的核心代码。
// 如果需要看源码,可以去 RxJava 的 GitHub 仓库下载。
public <R> Observable<R> lift(Operator<? extends R, ? super T> operator) {
    return Observable.create(new OnSubscribe<R>() {
        @Override
        public void call(Subscriber subscriber) {
            Subscriber newSubscriber = operator.call(subscriber);
            newSubscriber.onStart();
            onSubscribe.call(newSubscriber);
        }
    });
}

它生成一个新的<code>Observable</code>并返回,而且创建新<code>Observable</code>所用的参数<code>OnSubscribe</code>的回调方法<code>call()</code>
中实现竟然看起来和前面讲的<code>Observable.subscribe()</code>一样!然而它们并不一样哟~不一样的地方关键就在于第二行onSubscribe.call(subscriber)中的 onSubscribe所指代的对象不同

*subscribe()中的 onSubscribe指的是 Observable中的 onSubscribe对象,这个没有问题,但是 lift()之后的情况就复杂了点。
*当含有lift()时:
1、<code>lift()</code>创建了一个Observable后,加上之前原始的Observable,已经有两个Observable对象了.
2、同样,新Observable里的OnSubscribe加上原始的OnSubscribe,也有两个OnSubscribe;
3、当调用lift()后的subscribe()时,使用的是lift()所返回的新Observable,于是它被所触发的onSubscribe.call(subscriber)也是新Observable中的OnSubscribe;
4、新OnSubscribe的call()方法中的onSubscribe,就是指原始Observable中的OnSubscribe.在这个 call()方法里,新 OnSubscribe 利用 operator.call(subscriber)
生成了一个新的 Subscriber(Operator 就是在这里,通过自己的call() 方法将新 Subscriber 和原始 Subscriber 进行关联,并插入自己的『变换』代码以实现变换),然后利用这个新Subscriber 向原始 Observable进行订阅。

这样就实现了 lift()过程,有点像一种代理机制,通过事件拦截和处理实现事件序列的变换。

精简掉细节的话,也可以这么说:在 Observable 执行了 lift(Operator) 方法之后,会返回一个新的 Observable,这个新的Observable会像一个代理一样,负责接收原始的 Observable发出的事件,并在处理后发送给 Subscriber。
同时可以看图:

Paste_Image.png

此外,RxJava提供很多变化操作符如Buffer、FlatMap、Map、GroupBy、Scan、Window。想具体了解的查看这里。

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

推荐阅读更多精彩内容

  • 我从去年开始使用 RxJava ,到现在一年多了。今年加入了 Flipboard 后,看到 Flipboard 的...
    Jason_andy阅读 5,451评论 7 62
  • 前言我从去年开始使用 RxJava ,到现在一年多了。今年加入了 Flipboard 后,看到 Flipboard...
    占导zqq阅读 9,158评论 6 151
  • 文章转自:http://gank.io/post/560e15be2dca930e00da1083作者:扔物线在正...
    xpengb阅读 7,017评论 9 73
  • 亲爱的白先生: 微博看见你复活的消息 异常的兴奋啊 不过我也不想你复活 那样便没有很多人认识你 那样你便...
    很白的烨烨阅读 145评论 0 0
  • 第一次尝试画水彩人像,没有临摹,结果五官完全不会画,不得已只能搁置…… 不过总的来说,因为选了难度大的角度(真是活...
    乔原阅读 273评论 0 1