一篇不太一样的RxJava介绍(二):关于操作符背后的故事

前言: 上篇文章介绍了Observable这个类的来历。但是操作符是RxJava又一大优势。这篇文章我会介绍一下操作符背后的相关概念。
(读完这篇文章可能会引起身体强烈不适,甚至出现你以前懂操作符,读了之后反而不懂的情况。甚至这篇文章对你开发Android App不会有很大帮助,所以这篇文章需要谨慎阅读)

我们在了解操作符之前,首先要了解几个概念: Monad函数式编程。这里我会一一介绍他们,但是不会太详细,一篇文章肯定不能详细的介绍完这两个巨大的概念,甚至我自己都没有理解透彻这两个概念,但是这并不妨碍我们理解RxJava的操作符。

函数式编程

我们首先来说函数式编程,函数式编程的意义很简单。就是 用函数来编程。或者说,是用数学概念上的函数( mathematical functions )来编程。函数是两个集合之间的一种映射。
我们常常用 f:x -> y 这种形式来表示函数f是从X到Y的一种映射。
用我们熟悉的Kotlin语言来表示就是

    fun f(x:X):Y

但一般这种函数需要满足一下几个条件,我们才说这个函数是一个 Pure Function 也就是纯函数。

  1. 对应一个相同的输入值 x, 一定会获得一个相同的输出值 y。
  2. 在执行 f 的时候不会产生任何副作用

这里,我们又遇到了一个新名词,副作用。我们先来看维基百科对Side Effect的解释:

在计算机科学中,函数副作用指当调用函数时,除了返回函数值之外,还对主调用函数产生附加的影响。例如修改全局变量(函数外的变量)或修改参数。

也就是说,任何会改变外部状态的操作,都会被考虑为副作用,包括但不仅限于

  1. 对I/O的操作。例如读取文件或者在控制台输出,打印log。
  2. 修改外部变量,或者修改函数本身的参数。
  3. 抛出异常。
    等等。

Side Effect在函数式编程被认为是不好的东西。因为它太不可控了,比如常用的System.currentTimeMillis()方法。
我们每次调用这个方法,都会返回一个不同的值,这便是所谓的不可控。再比如readLine()函数,我们也无法知道他究竟会读取哪一行。

但是反过来,如果我们不是生活在“美好”的纯函数世界里。在我们的世界里,如果没有side effect,几乎做不了任何事。没有Side Effect我们甚至都不会接收到用户输入,因为用户的输入,比如屏幕点击都是一个Side Effect。为了解决这个问题,在Haskell(一种纯函数式编程语言)中,引入了Monad,来控制Side Effect。

Monad

我们说Side Effect虽然是不好的,但是是有用的。我们不希望消除Side Effect,我们更希望的是Side Effect在我们掌握之中,是可控的。所以引入Monad,来控制Side Effect。
Monad 在函数式编程中,有太多的教程,文章来解释。但是看了之后都云里雾里,甚至有人说过:

The curse of the monad is that once you get the epiphany, once you understand - "oh that's what it is" - you lose the ability to explain it to anybody.

Monad的诅咒就是一旦你理解他了,你就失去了向别人解释他的能力。

我不敢说这个诅咒在我这篇文章中消除了,我只能尽我所能,用一个Android开发者读得懂的语言尽力解释这个概念,所以我也在前言中提到了,这篇文章读后可能会引起严重不适。

So,言归正传,什么是Monad。

我们回到刚才的纯函数, 一个纯函数比如

f : x -> y

我们如何给他加入一个可控的Side Effect?
有一种做法便是,把Side Effect统统装进一个盒子里,和y一起当做输出值输出。
比如

f : x -> S y

S 代表了在输出y之前一系列Side Effect相关的操作。 但是这样的问题就是,我们如果连续进行好几个Side Effect操作。我们都要带着这个S,比如我们有两个函数f,g:

f : x -> S y
g : y -> S z

那么我们连续调用f,g之后,那结果就变成了:

f (g(x)) : x -> S(Sz)

这里Monad就要显示他的作用了。 很明显,我们需要一种“组合”的能力,将两个S结合成一个,我们更希望多个S可以结合成一个,比如这样:

f(g(x)) : x -> S z

一个Monad 我们简单的定义为有包含如下两个操作的盒子S:

  1. 一个进入盒子的操作(Haskell中的return) return: x -> S x
    在RxJava的世界中,更像是一系列产生Observable的操作符,比如create,just,fromXXX等等。比如:
    val x = 10
    Observable.just(x)
    // 这里我们进入了Monad的世界,而这个Monad是我们的Observable
  1. 一个"神秘"的运算bind(haskell中的==>)。 也就是我们结合的能力,他会接收一个函数 f: x -> M y 将两个带有Monad的函数连在一起。

Haskell的定义: (>>=) :: m x -> ( x -> m y) -> m y

我相信大家是看不懂的,我们用Java的语言来形容一下,我们知道Java中函数不是一等公民,不能直接当参数传给方法。我们只能用接口来模拟一个函数。
我们来定义我们的函数 function:

public interface Function<T,R>{
    R apply(T t)
}

T就是我们的输入,R就是我们的输出。(这个其实是Java 8 中的Function接口)。

而这个bind函数,就是接收一个函数f: x ->M y,然后自己生产出一个M y,我们暂时在Java世界中用Monad<X>来代表一个Monad。

public class Monad<X> {
    public Monad<Y> bind(Function<X,Monad<Y>> function) 
}

也就是,我们刚才所说的,结合的能力。我们通过接收一个 x -> M y 将我们的Monad<X>转换成了 Monad<Y>,而不是Monad<Monad<Y>>这样的嵌套操作。
但其实本质上,我们得到的Monad<Y>还是将我们本来的Monad<X>包裹在里面,只是形式上我们得到了Monad<Y>
这一部分用kotlin 可以更简洁的表达:


class Monad<X>

fun<X,Y> Monad<X>.bind(function:(X) -> Monad<Y>) :Monad<Y>

在上一篇文章中,我曾经说过

Collection可以通过高阶函数(High Oroder Function)进行组合,变换等等,所以作为集合之一的Observable也可以进行组合,变换。

但是其实这句话是错误的,因为在上一篇文章中,我们并没有Monad,函数式等等的知识,我们只能先这么理解。而给予Observable这个组合,变换能力的其实就是这个Monad。
结论1 :

Observable 是一个 monad

如果入门RxJava是从RxJava1 和 扔物线大佬的给 Android 开发者的 RxJava 详解这篇的话。 会知道RxJava 1中有一个
lift()操作符。是几乎所有操作符的“父”操作符,其实这也就是Monad中的bind的一个具体实现。也有人将flatMap理解为Monad中的bind,我个人认为是不对的。他们虽然签名是一致的,效果也是一样的。但是flatMap操作符在RxJava中的实现和其他操作符是非常不一样的。而lift()在RxJava 1.x 中就担任了所有操作符的抽象的工作。也就是我们说的接收一个 x-> Observable y 这样一个函数,来将Observable x 转换为 Observable y这样一个过程。而在RxJava2 中,由于性能问题,lift()操作符实现改为了直接继承Observable,来将lift的操作写到subscribeActual()来进行操作。这样虽然减少了性能损耗,但是正确的写一个操作符却变得更加困难一些。

当然,不是仅仅有return 和 bind 就可以是Monad,Monad 还需要满足如下三个规则:
这里我们用id(X) 来代表return

  1. 左单位元:

    id(X).bind(f:X -> Monad<Y>) = Monad<Y>

    也就是bind 在左边加上id这个函数,他获得的还是 bind的结果Monad<Y>本身。
    用RxJava 来表示就是

        Observable.just(1)
                .flatMap(new Function<Integer, ObservableSource<String>>() {
                    @Override
                    public ObservableSource<String> apply(Integer integer) throws Exception {
                        return Observable.just(integer.toString());
                    }
                })

  //这里在just之后flatMap的observable 和我们直接使用Observable.just("1")没有任何区别

  1. 右单位元:

    Monad(X).bind(id) = Monad<X>

    也就是 如果Monad<X>和 id 这个函数来进行结合,我们得到的还是Monad<X>
    用RxJava 来表示就是

        Observable observable = Observable.just(1)
                .flatMap(new Function<Integer, ObservableSource<Integer>>() {
                    @Override
                    public ObservableSource<Integer> apply(Integer integer) throws Exception {
                        return Observable.just(integer);
                    }
                })
    
    //这里进行过 flatMap 的 observable 和我们的Observable.just(1)没有任何区别
  1. 结合律:
Monad<X>.bind(function :X -> Monad<Y>).bind(function:Y -> Monad<Z>) 
    = Monad<X>.bind(function:x -> Monad<Y>.bind(function: Y -> Monad<Z>))

也就是,将后面两个Monad<Y>,Monad<Z>合并在一起,再和Monad<X>合并。和先合并,Monad<X>,Monad<Y>,在与Monad<Z>合并,效果是一样的。
用RxJava 来表示就是

        Observable observable1 = Observable.just(2)
                .flatMap(new Function<Integer, ObservableSource<String>>() {
                    @Override
                    public ObservableSource<String> apply(Integer integer) throws Exception {
                        return Observable.just(integer.toString());
                    }
                })
                .flatMap(new Function<String, ObservableSource<Double>>() {
                    @Override
                    public ObservableSource<Double> apply(String s) throws Exception {
                        return Observable.just(Double.valueOf(s));
                    }
                });

        Observable observable2 = Observable.just(2)
                .flatMap(new Function<Integer, ObservableSource<Double>>() {
                    @Override
                    public ObservableSource<Double> apply(Integer integer) throws Exception {
                        return Observable.just(integer.toString())
                                .flatMap(new Function<String, ObservableSource<Double>>() {
                                    @Override
                                    public ObservableSource<Double> apply(String s) throws Exception {
                                        return Observable.just(Double.valueOf(s));
                                    }
                                });
                    }
                });
    //这里 observable1 和 observable2 等价

遵守以上三个规则,并且拥有return/id 和 bind的“盒子”,我们就称之为一个Monad。我们在理解Monad之后,会发现我们身边很多东西,甚至每天都在用的一些东西,他就是Monad。
比如C#中的LINQ是Monad,Java 8新引入的CompletableFuture和Stream API是Monad, JavaScript中的Promise是Monad,RxJava中的Observable是Monad。
这也就解释了很多人在理解RxJava源码的时候,不理解为什么 Observable 操作符要写成这种 Observable套着Observable。最终互相通知的形式。
如:(这里为了简化我们使用Kotlin来写)

        Observable.just(1, 2, 3, 4)
            .map{x -> x +1}
            .filter { x -> x >3 }
            .flatMap { x -> Observable.just(x,x+2) }

这其实生成的Observable是 ObservableFlatMap(ObservableFilter(ObservableMap(ObseravbleJust(1,2,3,4)))) 这样一个一层层嵌套的Observable盒子。而赋予其嵌套能力,并将其省略为仅仅一个Observable强大力量的便是Monad。
所以我们得出一个结论2

Observable的操作符 Monad中 bind 的一个具体实现形式。

而这个结论并不适合所有操作符,有一些特殊操作符会从Monad中跳出返回我们正常的Java/Kotlin世界。比如Subscribe,blockingFirst(),forEach()等等。
这些是我们跳出Monad/Observable世界的出口。

总结:
这篇我主要介绍了函数是编程和Monad的概念,着重介绍了Monad和Observable紧密的关系。个人认为如果对函数式编程不感兴趣,对Monad的意义不必太过纠结,只需将其理解为一种对集合进行组装变换的一种解决方案即可。

参考文献(部分链接可能需要梯子)

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

推荐阅读更多精彩内容