Android单向数据流——Side Effect

前言

背景知识:
Android单向数据流——MvRx核心源码解析
Unidirectional data flow on Android using Kotlin

本文讨论一个问题,Android单向数据流中的Side Effect(副作用),看看MvRx是如何实现Side Effect的,经典的单向数据流Redux又是如何实现Side Effect的?为什么说MvRx是简化的Redux,MvRx又有哪些问题呢?

Side Effect

什么是Side Effect?我们知道,单向数据流的核心是StateStore中的Reducer,Reducer负责根据接收到的Event/Action/Intent/Wish,别管怎么叫吧,总之根据接收到的数据以及当前的State生成一个新的State,这样才能驱动整个单向数据流流动起来。

这里有两层隐藏的含义:

  1. State必须是“不可变”(Immutable)的
  2. Reducer执行的函数必须是“纯函数”(pure function)

如果State是可变的,那么我们很可能无意间就直接修改了State,那么新旧State也就无从对比,这会降低State的传输效率;更关键的一点是,这会引起多线程修改State的并发问题,大大增加State管理的复杂度。
如果Reducer执行的函数不是纯函数,也就是说这些函数带有副作用,那么单向数据流就无从谈起了,我们可以通过“副作用”去修改State,Reducer还有什么存在的意义。
所以说,“不可变性”和“纯函数”是Reducer的应有之意。

Reducer体现的其实就是函数式编程的思想,但是函数式编程总是要解决的一个问题就是如何处理“副作用”。“副作用”是无处不在的,典型的副作用就是IO操作(文件读写、数据库操作、网络请求),显然这些都是避免不了的,函数式编程给出的方案是,将“副作用”打包成函数去执行。
我们不去管这些概念,来看看单向数据流中的副作用如何处理。理想状态下,Action1会生成State1,Action2会生成State2,没有副作用,但这是不可能的,假设Action1是网络请求,Action2是数据库查询,Action1、Action2都是无法直接执行的,但是我们可以这么做,Action1触发Side Effect1,Action2触发Side Effect2,Side Effect1/2去执行这些副作用,执行结束后,返回Action1',Action2',这时候Action1',Action2'就是纯函数了,可以在Reducer中继续执行。

以上就是典型的Redux单向数据流的流程,总结起来就是,只有Action可以改变State,而Action可能会触发Side Effect,Side Effect执行完成后再次发送Action到Reducer。

MvRx Side Effect

Side Effect被称为副作用有点名不符实了,对于Reducer而言的确是Side Effect,但对于业务逻辑而言,这些Side Effect才是真正的“作用”,而Reducer中进行的State更新才是所谓的“副作用”。Side Effect(例如网络请求、数据库操作)对于应用而言是如此的重要,把它看作是Action触发的副作用有点主次颠倒了,因此MvRx并没有采用经典的Redux模型,而是采用了简化的Redux:

  1. 没有所谓的Action,既然Reducer的核心就是f(State)=State,那就直接向Reducer提供State.()->State类型的元素,省略从Action到f(State)=State的映射。并且Debug模式下,State.()->State会连续运行两遍,尽量保证State.()->State是纯函数。
  2. 没有所谓的Side Effect,对于明确要进行的网络请求、数据库操作、文件读写就先进行这些Side Effect(使用RxJava Observable进行包装),拿到结果后再提供给Reducer;还可以使用withState的方式,先获取State状态,再根据State触发相应的Side Effect。总之,不需要通过Action来触发Side Effect。

总结起来就是,Action、Side Effect与f(State)=State的统一,省略Action、Side Effect这些概念,突出Reducer的核心功能。这么做或许会降低一些复用性,例如,如果存在Action和Side Effect,我们可以建立起Action、Side Effect与f(State)=State多对多的对应关系,增加Action、Side Effect、f(State)=State复用的可能性,抽象层次更高,解耦更加彻底。但是,MvRx以更低的抽象层次,换来了概念上的简化,逻辑上的连续,或许会牺牲一些复用性,但是现实中,真的没有那么多可复用的东西,逻辑上的连续往往比所谓的解耦更加重要。

MvRx的问题

就像上一篇文章说的那样,每个StateStore都会新建一个线程用于执行reducer,也就是上图中的flushQueue方法,这避免了多线程状态下State的同步。单线程执行reducer没有问题,但是否每一个reducer都需要一个单独的新线程呢?正如我们前面说的那样,MvRx没有Action、Side Effect这些概念,flushQueue执行的就是一些State.()->State纯函数,这些纯函数一般而言就是Kotlin Data Class下的copy方法,众所周知,copy方法实现的是浅拷贝,一般不会有什么性能问题。因此,我认为为每一个StateStore分配一个新的线程来执行flushQueue有点多余,可以让所有StateStore共用同一个线程,这样既保证了单线程执行flushQueue,又减少了线程创建销毁的开销。(我给MvRx提过issue,但是他们并不接受)


以上是我认为MvRx中存在的一个问题,我认为MvRx还存在以下一些问题:

  1. MvRxViewModel实现依赖注入不友好,需要在每个MvRxViewModelcompanion object中实现特定的接口,简直累死。根本问题在于MvRx没有提供自定义ModelViewFactory的方式,当然,MvRx是为了保证StateMvRxViewModel的一致性,因为初始State对于MvRxViewModel而言是很重要的。
  2. 过多的反射,初始State的创建必须通过反射,即使在大多数情况下,我们可以为初始State提供默认值;通过@PersistState保存State也必须通过反射(使用简单但是效率低),不如使用ViewModel SavedState,ViewModel SavedState使用也很简单,并且不使用反射,效率更高(之所以这样也是有原因的,因为先有的@PersistState,然后才有的ViewModel SavedState)。MvRx使用反射的地方太多了,有些是合理的,有些我觉得有点过度了。

虽说MvRx源码比较简单,想自己修改这些问题也不是什么难事,但是与官方版本脱节也不是一件好事。

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