RxJava Map操作详解

2016-06-06

  RxJava是最近两年火起来的一个框架,核心是异步,但是对于我来说印象最深的是响应式编程的思想。最近刚好想把自己的项目改成用RxJava实现,所以就研究了下。抛物线和大头鬼两位大牛也讲解的很详细和形象,其实RxJava里除了这种响应式的编程思想不太好理解外,操作符也是比较难理解的一部分。响应式编程思想不是三言两语就能讲清楚,想学习的人也不是通过看几遍blog就能学会的。我这里主要是讲操符,通过分解的方式一步一步带领大家看着到底是怎么回事,就以常用的map为例。

首先看一段伪代码:

Observable.create(new Observable.Onsubscrible(){  //------------ Observable one
  @override
  public object call(…subscriber){  //----------------------call one
     ……
     subscriber.onNext();  //-------------- onNext one
     …….
}
}).map(new Func1(){ // ----------------------map 操作后返回的为Observable two
  @override
  Public object call(…){   //-----------------------------call two
     …….
}

}).subscrible(new Subscriber(){
   @override
   public void onCompleted(){}
   @override
   Public void onError(){}

   @override
   Public void onNext(){  //-------------------------onNext two
     …….
    }
});

我将剖析上面的一段RxJava代码。为了一步一步的把问题描述清楚,我们先把上面的代码简化如下,定义observableTemp变量:

observableTemp.subscrible(new Subscriber (){…..});

其中ObservableTemp为:

observableTemp =
observable.create(new Observable.Onsubscrible(){  //------------ Observable one
  @override
  public object call(…subscriber){  //----------------------call one
     ……
     subscriber.onNext();  //-------------- onNext one
     …….
}
}).map(new Func1(){ // ----------------------map 操作后返回的为Observable two
  @override
  Public object call(…){   //-----------------------------call two
     …….
}

});

   

第一步,我们先解析subscrible到底做了什么

  hook.onSubscribeStart(observable, observable.onSubscribe).call(subscriber);

这是整个subscrible方法的核心。onSubscribeStart返回的是observable.onSubscribe,而observable.onSubscribe是传入参数,为this对象。所以可以把subcrible方法的主要功能简化为:

 ObservableTemp. onSubscribe.call(new Subscriber (){….});

举例说明下:如果

ObservableTemp =

Observable.create(new Observable.Onsubscrible(){

@override

public object call(…subscriber){  //---------------1-1

subscriber.onNext();

}

})

那么ObservableTemp. onSubscribe.call(new Subscriber (){….});分解如下:

ObservableTemp. onSubscrib = Observable.Onsubscrible()

这里的1-1传入值按照我们刚才简化后形式,应该是new Subscriber ()。那么整个就成了:

new Observable.Onsubscrible(){

@override

public object call(…subscriber){

……

subscriber.onNext();

……..

}

}.call(new Subscriber (){….});

这里的call就是调用的@override的call方法,所以这里进一步分解:

public object call(){

……

new Subscriber (){….}.onNext();

……..

}

看明白了吗?其实整个一大段代码只是执行了Subscriber里的onNext方法。

相信看到这里已经明白了subscribe方法的作用了。但是如果把ObservableTemp展开呢?也就是还原到开头的那段代码,又干了什么事情呢?请看第二步的分解

  

第二步,map到底做了什么

  我们已经知道ObservableTemp. onSubscribe.call(new Subscriber (){….});其实是通过ObservableTemp里的Onsubscrible.call方法直接调用Subscriber的onNext方法。那么加入map后,到底是怎么一种调用关系呢?Map字面的意思应该是个映射操作,那到底是不是字面上的映射?如果是映射,到底谁和谁之间的映射呢?

为了方便说明,我们还是把开头第一段代码做个简化,简化为:

ObservableTemp.map(new Func1(){…}).subcrible(new Subscriber(){…});

其中ObservableTemp 就是一个简单的对象创建和赋值过程了:

Observable.create(new Observable.Onsubscrible(){
  @override
  public object call(…subscriber){
     ……
     subscriber.onNext();
     …….
}
}); // 没有什么理解上的难度

map的操作代码是:

lift(new OperatorMap<T, R>(func));

lift是RxJava里核心,几乎大部分Observable操作都是需要用到lift方法。A.lift(Operator op)返回的是一个新的Observable,新Observable中的OnSubscribe为OnSubscribeLift。

ObservableTemp.map(new Func1(){…}).subcrible(new Subscriber(){…});

就变成了:

new Observable(new OnSubscribeLift(){
@override
  Public object call(…){
     …….
}
}).subcrible(new Subscriber(){…}); // ------------------------- 2-2

是不是很眼熟,Yes, 就是第一步里简化的步骤。那么这里关键的部分来了,OnSubscribeLift里的call做了什么,如果我们把这个搞清楚了,那么map就自然而然的就清楚了。

OnSubscribeLift里的call主要有两行代码:

Subscriber<? super T> st = hook.onLift(operator).call(o);
parent.call(st);

这里的operator,o ,parent以及 st到底是什么呢?

因为这里的o是OnSubscribeLift里call方法传入的参数,由2-2可以知道,o其实代表的是new Subscriber()。

再看operator,由代码可以看出:

public final <R> Observable<R> map(Func1<? super T, ? extends R> func) {
        return lift(new OperatorMap<T, R>(func));
    }

这里的operator就是指经过OperatorMap包装过的Operator对象:new OperatorMap<T, R>(new Func1());而hook.onLift(operator)返回的就是operator,那么上面两句代码可以看成:

Subscriber<? super T> st = (new OperatorMap<T, R>(new Func1())).call(o);
parent.call(st);

OperatorMap里的call做了三件事情:

MapSubscriber<T, R> parent = new MapSubscriber<T, R>(o, transformer);
o.add(parent);
return parent;

把 new subscriber()和new Func1()(就是transformer对象)通过MapSubscriber封装起来。然后新的Subcriber(MapSubscriber对象)放入订阅列表中,以便最后一起把订阅对象释放。同时返回新的Subcriber。

根据上述的,

Subscriber<? super T> st = (new OperatorMap<T, R>(new Func1())).call(o);
parent.call(st);

可以看成:

Subscriber<? super T> st  = new MapSubscriber<T, R>(o, transformer);
parent.call(st);

而parent是传入的参数,指ObservableTemp.onSubscribe,也就是开头代码中的Observable one。

进一步上述两行代码可以看成:

Subscriber<? super T> st  = new MapSubscriber<T, R>(o, transformer);
(new Observable.Onsubscrible(){  //------------ Observable one
  @override
  public object call(…subscriber){  //----------------------call one
     ……
     subscriber.onNext();  //-------------- onNext one
     …….
}
}).call(st);

这里的override里的call参数为new MapSubscriber<T, R>(o, transformer); new MapSubscriber<T, R>(o, transformer).onNext主要代码为:

@Override
 public void onNext(T t) {
    result = mapper.call(t);
    actual.onNext(result);
 }

Mapper就是transformer对象,也就是new Func1(),而actual就是new subscriber()。也就是说:把Observable one里的类型经过new Func1().call的方法转换成另外一个Subscriber,最后调用new Subscriber的onNext方法。

所以map整个过程就清楚了:
A.map(B).subscribe(C) 就是:

先通过map方法,把A中想要转换的数据通过调用B里的call方法进行转换,最后把转换过的数据用C里的onNext方法进行处理。

我们抛开onCompleted和onError方法,

A.map(B).subscribe(C)  <==> 
C.onNext (B.call(A.onNext()))

时间: 2024-12-24 18:51:35

RxJava Map操作详解的相关文章

STL中的map用法详解

STL中map用法详解 说明:如果你具备一定的C++ template知识,即使你没有接触过STL,这个文章你也应该可能较轻易的看懂.本人水平有限,不当之处,望大家辅正. 一.map概述 map是STL的一个关联容器,它提供一对一(其中第一个可以称为关键字,每个关键字只能在map中出现一次,第二个可能称为该关键字的值)的数据处理能力,由于这个特性,它完成有可能在我们处理一对一数据的时候,在编程上提供快速通道.这里说下map内部数据的组织,map内部自建一颗红黑树(一种非严格意义上的平衡二叉树),

Windows DIB文件操作详解-4.使用DIB Section

前面讲了为了提高DIB的显示性能和效率,我们将DIB转换成DDB,但是这又遇到一个问题,如果我想操作DIB的数据的话,显然是不能使用DDB:一是因为DIB转DDB时发生了颜色转换,再就是DDB无法直接提取指定像素点的数据.那么我们怎么办呢,Windows使用一种折中的方式来达到这一目标(既提高了显示效率和性能,又可以直接操作像素点). 1.DIB Section存储和显示 Windows使用DIB块(DIB Section)来存储DIB数据,其内存结构示意图如下 其实,和我们自己读入DIB数据到

Linq实战 之 DataSet操作详解

Linq实战 之 DataSet操作详解  一:linq to Ado.Net 1. linq为什么要扩展ado.net,原因在于给既有代码增加福利.FCL中在ado.net上扩展了一些方法. 简单一点的说: 就是在DatTable 和 DataRow 上面做了一些扩展. 二:扩展方法一览 1. AsEnumerable 2. Field 三:扩展类一览 DataTableExtensions 扩展 => public static EnumerableRowCollection<DataRo

DNS基本概念及操作详解----------------转载

DNS基本概念及操作详解 目录: 1.DNS协议 2.DNS查询 2.1递归查询 2.2跌代查询 2.3反向查询 3.域维护 3.1全量AXFR传输 3.2增量IXFR传输 3.3通过NOTIFY 3.4动态更新 4.DNS安全 在很多人看来,DNS只是为外部提供DNS解析服务(我以前也是这么认为的,直到膝盖中了一箭),但作为互联网的基础设施,DNS远没有想象的那么简单.如果你没有听说过DNS查询.反向解析.zone传输.动态更新.DNS安全,那你可以从本文中得到关于他们的最简明的诠释. 一.

Python对Excel操作详解

  Python对Excel操作详解 文档摘要: 本文档主要介绍如何通过python对office excel进行读写操作,使用了xlrd.xlwt和xlutils模块.另外还演示了如何通过Tcl  tcom包对excel操作. 关键字: Python.Excel.xlrd.xlwt.xlutils.TCl.tcom     1 Python简介 Python是一种面向对象.直译式电脑编程语言,具有近二十年的发展历史,成熟且稳定.它包含了一组完善而且容易理解的标准库,能够轻松完成很多常见的任务.

Yii 框架里数据库操作详解-[增加、查询、更新、删除的方法 &#39;AR模式&#39;]

public function getMinLimit () {        $sql = "...";        $result = yii::app()->db->createCommand($sql);        $query = $result->queryAll();         return array (                $query [0] ['max'],         );    } $connection=Yii::

Linq实战 之 Linq to Sql及Entity Framework操作详解

Linq实战 之 Linq to Sql及Entity Framework操作详解 一:linq to db的框架 1. linq to sql 2. linq to ado.net entity framework linq to sql是一个团队 ef 是一个团队... linq to sql => ef 团队. linq to sql 适合一些小型的项目 => sqlserver ef 适合中形的项目,而且可以支持 sqllite,mysql,sqlserver 掌柜的项目开发中:使用的

笔记-[1]-DOM的节点操作详解.

DOM:文档对象模型 操作DOM基本就是操作DOM的元素节点. 节点的属性: 1:objElement.childNodes  :获取该元素对象的的节点集合,有length长度属性,在标准浏览器下使用,会辩认文本节点的节点,一般不用,有其他的更好的属性(children) 2:objElement.children :     获取该元素对象的的节点集合,有length长度属性,在ie8下和其他标准浏览器兼容,只获取元素节点. 3:obj.nodeType   :获取对象的节点类型,1为元素节点

Windows DIB文件操作详解-1.DIB的读入、保存和显示

DIB(设备无关位图)是存储在磁盘上的位图文件,可以从磁盘读到内存中或从内存保存到磁盘上,它的磁盘文件结构是标准化的,在Linux.Unix及Windows上都可以以同样效果显示.位图是最接近硬件的图像格式,Windows显示的核心是位图,它的SDK API专门提供了一组用于操作DIB文件的函数.但是由于这样或那样的原因,高效合理的使用这些DIB API是需要了解不少历史和使用背景的,在这里我抽茧剥丝介绍和演示DIB的使用,相信对你更好的使用DIB文件有帮助,由于DIB函数比较多,这里分为三部分