简评: Redux 是一个用于应用程序状态管理的开源JavaScript库,其核心是通过可管理和控制的状态来描述一个系统。这意味着其思想其实是可以应用于任何类型应用的开发的,包括移动应用。

Redux 架构基于一个严格的单向数据流,应用中的所有数据都是通过组件在一个方向上流动。Redux 希望确保应用的视图是根据确定的状态来呈现的。意思就是,在任何时间点,你应用的状态总是确定、有效的,并且可以转换到另一个可预测和有效的状态。而 UI 将根据所处的状态来进行呈现。

关于 Redux 在网上已经有很多相关的资料,这里就只介绍下 Redux 核心的三个组件:

1.Store:保存应用的状态并提供一些帮助方法来存取状态,分发状态以及注册监听。

2.Actions:简单说 Actions 就是事件,包含要传递给 store 的信息,表明我们希望怎样改变应用的状态。比如,定义如下的一个 action:

data class AddTodoAction(val text: String)

由 store 来进行分发:

store.dispatch(AddTodoAction("Write blog post"))

3.Reducers:进行状态的转变。类似这样:

fun reduce(oldState: AppState, action: Action) : AppState {
return when (action) {
is AddToDoAction -> {
oldState.copy(todo = ...)
}
else -> oldState
}
}

介绍完核心组件,下面来看一下它们是怎么组合到一起的:

Redux 的流程很简单,你的应用根据当前状态呈现 UI,用户的交互触发 action,并交给 reducer 来更新状态。

最近,作者在一个还挺大的项目上试了下 Redux 架构,所以这里就分享下从中总结的一些经验。

1. 应用里最好不要有多个 store

针对不同模块有不同的 store 似乎是个不错的主意,但从上面的图可以看到每个 store 和其数据流是一个闭环系统,这就使得不同 store 之间的状态难以同步。这样你就通常需要在一个状态的变更响应中去进行另一个 store 的 action 分发,而这很容易造成无限循环。

另一个原因是多 store 的架构是非常僵化的,难以灵活的改动。

更好的做法是维护一个包含多个子状态的全局应用状态,由一个 store 来表示:

data class AppState(val LoginState,
val HomeScreenState,
val GridState )

2. 保持应用的状态层级尽可能少

因为 Redux 中 state 是不可变的,因此深层次嵌套的 state 会产生很多的样板代码,并且难以更新。比如,考虑下面的一组数据模型:

data class State(val sections: List<Section>)

open class Section(val articles: List<Article>)
class Home(articles: List<Article>) : Section(articles)
class Discover(articles: List<Article>) : Section(articles) class Article

实例化和更新状态对象:

val state = State(sections = listOf(
Home(listOf(article1, article2)),
Discover(listOf(article1, article2))))

即使是用了 Kotlin 的 copy 机制,更新深层嵌套的属性(比如上面的 Article)也是非常单调乏味的:

val newHome = Home(listOf(newArticle, state.sections[0].articles[1]))
state.copy(sections = listOf(newHome, state.sections[1]))

3. Reducers 只是纯函数

Reduce 的作用只是处理 action 并返回新的 state 到 store 的,需要保证相同的输入总会得到一样的输出。Reduce 自身不应该有状态和执行任何额外工作,而只是做状态转换。

class Reducer {
fun reduce(state: State, action: Action) : State {
...
}
}

如果你需要响应某个 action,并执行一些操作,那应该考虑使用 Middleware。

4. 只用 Kotlin

Redux 很大部分受到 Flux 的启发,而关于 Flux 最常见的抱怨就是需要写一大堆的样板代码。而所选择的语言很大程度会决定你管理样板代码的便利性。

Kotlin 中类似 data class,when 语句之类的特性,能让你的代码清晰很多。例如,在 Reducer 中匹配 action 时,可以选择用 instanceof 方法实现。

if (action instanceof AddTodoAction) {
return reduceAddTodoAction(oldState, action);
} else if (action instanceof RemoveTodoAction) {
return reduceRemoveTodoAction(oldState, action);
} else if (...) {
...
}
return oldState;

当 action 很多时,这种写法就很痛苦了。如果用 Kotlin 就是这样的:

return when (action) {
is AddTodoAction -> reduceAddTodoAction(oldState, action)
is RemoveTodoAction -> reduceRemoveTodoAction(oldState, action)
else -> oldState
}

结论

虽然,Redux 主要是被用于 Web 应用开发,但其思想我们还是可以学习并引入到 Android 中。但 Redux 也不是「银弹」,事实上也没有什么架构是,其在 Android 上的应用还很新,但我们还是很希望能看到它的逐渐成熟。

如果你对 Redux 在 Android 上运用有兴趣,可以看看 ReduxReKotlin 这两个库。


原文链接: Lessons learned implementing Redux on Android - Pusher Blog

推荐阅读:Touch Bar 废物利用系列 | 在触控栏上显示 Dock 应用图标

在 Android 中实现 Redux 的一点经验的更多相关文章

  1. Android中MVC、MVP、MVVM具体解释

    前言 今天有时间就刚好有想写关于这几个名词.对于我来说.事实上这么多名词.思想归根究竟就是要依据项目实际.人员配置来做合理优化,既不能纸上谈兵.又不能畏惧不前.那么合理分阶段架构和完好代码才是关键,本 ...

  2. Android中怎样做到自己定义的广播仅仅能有指定的app接收

    今天没吊事.又去面试了,详细哪家公司就不说了,由于我在之前的blog中注明了那些家公司的名字,结果人家给我私信说我泄露他们的题目.好吧,我错了... 事实上当我们已经在工作的时候.我们能够在空暇的时间 ...

  3. Android开发的16条小经验总结

    Android开发的16条小经验总结,希望对各位搞Android开发的朋友有所帮助. 1. TextView中的getTextSize返回值是以像素(px)为单位的, 而setTextSize()是以 ...

  4. 谈谈Android中的Rect类——奇葩的思维

    最近在工作中遇到了一些问题,总结下来就是Android中Rect这个类造成的.不得不说,不知道Android SDK的开发人员是怎么想的, 这个类设计的太奇葩了.首先介绍一下Rect类:Rect类主要 ...

  5. Android中jsoup的混淆规则【转】

    Android中jsoup的混淆规则版权声明:转载必须注明本文转自严振杰的博客:http://blog.yanzhenjie.com 说实话这篇文章的标题和内容我觉得很水,所以读者们要是也觉得这篇文章 ...

  6. Android中如何做到自定义的广播只能有指定的app接收

    今天没吊事,又去面试了,具体哪家公司就不说了,因为我在之前的blog中注明了那些家公司的名字,结果人家给我私信说我泄露他们的题目,好吧,我错了...其实当我们已经在工作的时候,我们可以在空闲的时间去面 ...

  7. (转)谈谈Android中的Rect类——奇葩的思维

    最近在工作中遇到了一些问题,总结下来就是Android中Rect这个类造成的.不得不说,不知道Android SDK的开发人员是怎么想的, 这个类设计的太奇葩了.首先介绍一下Rect类:Rect类主要 ...

  8. Android开发学习之路-Android中使用RxJava

    RxJava的核心内容很简单,就是进行异步操作.类似于Handler和AsyncTask的功能,但是在代码结构上不同. RxJava使用了观察者模式和建造者模式中的链式调用(类似于C#的LINQ). ...

  9. Android中矢量动画

    Android中矢量动画 Android中用<path> 标签来创建SVG,就好比控制着一支画笔,从一点到一点,动一条线. <path> 标签 支持一下属性 M = (Mx, ...

随机推荐

  1. on namespace ceilometer.$cmd failed: Authentication failed. 问题处理方案

    on namespace ceilometer.$cmd failed: Authentication failed. UserNotFound: Could not find user ceilom ...

  2. GCC 9.2 2019年8月12日 出炉啦

    GNU 2019-08-12 发布了 GCC 9.2https://gcc.gnu.org/onlinedocs/9.2.0/ 有详细的说明 MinGW 上可用的 GCC 9.2 版本下载地址 [ m ...

  3. json字符串转map、json数组演示

    公司项目用的IBM封装的json解析,此处采用阿里的fastjson进行演示,代码如下: package com.alphajuns.test; import com.alibaba.fastjson ...

  4. 基于.net 4.0框架的Cipher演示程序

    using System.Text; namespace Cipher.Algorithm { static class Caesar { static public string Encrypt(s ...

  5. java:Linux(简单命令,远程ssh使用hostname访问,.免密钥登录配置)

    1.临时关闭防火墙: service  iptables  stop 临时开启防火墙: service  iptables  start 查看防火墙状态: service  iptables  sta ...

  6. Day03:日期操作 / 集合框架(上)

    日期操作 Java中的时间 · Java中的时间使用标准类库的Date类表示,是用距离一个固定时间点的毫秒数(可正可负,long类型)表达一个特定的时间点: · 固定的时间点叫纪元(epoch),是U ...

  7. 增加游戏受众:在英特尔核显上运行MR游戏

    为了获得良好的用户体验,通常VR游戏需要稳定在每秒90帧,这意味着整个场景只有11.1毫秒的时间来进行左右眼渲染,这就是为何VR游戏需要配备高端的独立显卡.现在来看看我们如何使用最新版英特尔图形性能分 ...

  8. C#编程 Excel操作

    使用OLEDB操作Excel,关于OLEDB介绍参考http://www.cnblogs.com/moss_tan_jun/archive/2012/07/28/2612889.html 连接字符串: ...

  9. LeetCode.1071-字符串最大公约数(Greatest Common Divisor of Strings)

    这是小川的第391次更新,第421篇原创 01 看题和准备 今天介绍的是LeetCode算法题中Easy级别的第253题(顺位题号是1071).对于字符串S和T,当且仅当S = T + ... + T ...

  10. kean的博客今天开通了,happy 一下

    希望以后可以日日勤勉,孜孜不倦的记录我的一生!