用了几个月的redux,现在回过来总结一下。

  刚开始用的时候遇到一个比较大的疑问,就是如何设计redux的store中的state树,这应该是我在使用redux中最大的一个疑问,阻挡了我前进的脚步,经过一段时间的研究,我这里给大家用非常通俗易懂的方式说一说我的疑问,以及我是如何解决的。

  当初主要的疑问是:

1.state树是按照页面划分

2.还是按照数据库中的表划分

   第一种就是把每个页面当成一个业务模块,实际上这也是比较流行的一种做法,有很多公司是这么做的,比如你可以像下面这样初始化你的store

module.exports = function() {
return {
common: {
isFetching: false
},
   profile: {
   },
list: {
},
   edit: {
   },
home: {
}
}
}

  这种方式的优点就是模块与模块之间互相独立,不会相互影响,每个页面维护自己的reducer,并且只在store中存入该页面展示或者变化的最小数据,使用起来很方便,不太需要关心其他模块的缓存数据,比较符合redux设计的初衷(并不是用来做一个前端数据库),但这种方法有个致命的缺点,就是store中数据的同步问题和冗余问题。假如一个列表页面的数据依赖于另一个编辑页面的操作,在编辑页面,改变了数据,这时候数据库里面的数据已经被改变了,但这时候列表页面和编辑页面的state是独立的,列表页面state节点下的数据并没有改变,退回到列表展示页,数据并不会发生变化,这就有问题了,而且同一份数据出现在多个节点下,也会有数据冗余的问题。

  要让依赖的列表页面也产生同步的更新,这里有四种做法:

  1.每次进入列表页面去数据库刷新store(如果你做的是一个用户端App显然不会这么干,只可能在第一次进入的时候才会载入数据,第二次应该就不会发请求了,立即从store中拿出缓存数据,否则性能和体验太差,PC端后台倒是无所谓)

  2.编辑完保存后dispatch一个或多个action,去改变所有依赖到这条数据的页面的state(相对还是比较可靠的办法,前提是依赖的页面很少,如果是一个庞大的系统,每次新增加一个页面,你就需要去修改派发action的代码,太麻烦了,除非建立一个完备的事件系统,所有依赖到这条数据的页面都监听一个事件,只要一编辑,就触发所有的事件,更新自己的state节点),如果是这种情况,可以考虑看一看 rxjs

  3.编辑完后除了更新数据库中的数据,在前端这边什么都不做,返回列表页,先取出缓存数据展示,等到页面DidMount之后发一个请求,不要出loading图,检查一下数据是否被更新过,如果是,拉取最新数据并更新store,如果不是,那就不变

  4.列表页和编辑页共用一个state数据节点,列表页只存必要显示的字段,点击进入编辑页,根据列表页的记录id读取更详细的信息,编辑保存完成后,更新到这个state节点(这种方法其实采用的就是按数据库表划分的办法,统一数据源,关于统一数据源格式,可以看一看 normalizr )

  注意:没有人告诉过你,用了redux必须通篇要用store中的state,redux作者也没有这么讲过,比较好的做法是结合react组件自己的state和store中的state一起使用。

   第二种就是按照数据库中的表划分,比如你可以像下面这样初始化你的store

module.exports = function() {
return {
common: {
isFetching: false
},
  companies: {
},
users: {
},
events: {
}
}
}

  这种方法可能比较适合单页应用,比如(trello,coding.net),把store当成数据库使,增删改查都维护这一个store,作为前端数据源,保证前端数据源和后端数据库同步,并且保证了操作的及时反馈,适合对用户体验要求较高的场景,缺点就是前端不像后端,除了model数据,还有业务数据,临时状态数据,不能简单的用数据库的思路去设计store,如果是这样,那么为什么不用前端数据库呢?为啥还要用redux来多此一举,redux本质上是用来管理数据状态的。

  所以,我认为还是按照页面来划分,但应该把第二种方法的思路结合进来。

  在此之前,先要认清哪些数据应该被放到store,如果是临时数据,不需要缓存的,我们应该用react自己的state去维护状态,而不是丢进store。如果该数据需要缓存,但只被一个页面依赖了,那么放进该页面节点下的state。如果该数据需要缓存,被多个页面依赖了,那么我们应该将它提出来,放到一个公共state节点下。

  redux作者的意思大概就是,对于store中的state树,躺着用,站着用,跪着用,想怎么用怎么用,并没有明确规定必须怎么用,只要自己用的舒服,解决你自己的特殊场景就可以了。

如何优雅的设计Redux中的Store的更多相关文章

  1. 通过三张图了解Redux中的重要概念

    上周利用业余的时间看了看Redux,刚开始有点不适应,一下在有了Action.Reducer.Store和Middleware这么多新的概念. 经过一些了解之后,发现Redux的单向数据里的模式还是比 ...

  2. Redux中的重要概念

    Action/Reducer/Store 首先,先看看第一张图,图中展示了Redux的单向数据流,以及Action.Reducer和Store这三个核心概念. 下面就围绕上图,非别介绍Action.R ...

  3. 如何优雅的设计React组件

    如何优雅的设计 React 组件 如今的 web 前端已被 React.Vue 和 Angular 三分天下,一统江山十几年的 jQuery 显然已经很难满足现在的开发模式.那么,为什么大家会觉得 j ...

  4. Redux 中的CombineReducer的函数详解

    combineReducers(reducers) 随着应用变得复杂,需要对 reducer 函数 进行拆分,拆分后的每一块独立负责管理 state 的一部分. combineReducers 辅助函 ...

  5. 关于props和state以及redux中的state

    React的数据模型分为共有数据和私有数据,共有数据可以在组件间进行传递,私有数据为当前组件私有.共有数据在React中使用props对象来调用,它包含标签所有的属性名称和属性值,props对象有三个 ...

  6. 25.redux回顾,redux中的action函数异步

    回顾:Redux: 类似于 Vuex 概念:store/reducer/action action:动作 {type,.....} 一定要有type 其他属性不做限制 reducer:通过计算产生st ...

  7. 在react/redux中使用Immutable

    在redux中使用Immutable 1.什么是Immutable? Immutable是一旦创建,就不能被更改的数据. 对Immutable对象的任何修改或添加删除操作都会返回一个新的Immutab ...

  8. 在 react 项目里如何配合immutable在redux中使用

    一.reducer文件的处理 先安装 immutable 与 redux-immutable yarn add immutable redux-immutable 我们可能会在很多地方定义子树,这就需 ...

  9. 清晰理解redux中的

    首先需要明白 Redux 的单一状态树的概念,所谓的单一状态树,就是指“所有的 state 都以一个对象树的形式储存在一个单一的 store 中.” 比如我们有这么一个状态树(或者你叫它状态对象也行) ...

随机推荐

  1. mysql utf8mb4与emoji表情

    一 什么是Emoji emoji就是表情符号:词义来自日语(えもじ,e-moji,moji在日语中的含义是字符) 表情符号现已普遍应用于手机短信和网络聊天软件. emoji表情符号,在外国的手机短信里 ...

  2. Zend Server的WebAPI焦点:异步操作

    Zend Server作为Web应用服务器,在使用时,涉及连接的基本步骤,其中包括许多可用的WebAPI行为. WebAPI提供一些需要时间才能完成的操作,包括很复杂的或依靠外部实体(如远程服务器)才 ...

  3. 水晶报表在vs2010 WPF环境下的尝试

    原文:水晶报表在vs2010 WPF环境下的尝试 由于VS2010没有集成水晶报表组件,尝试前必须先安装 水晶报表 for VS2010,若机器未安装的可点击这里>>>下载安装 新建 ...

  4. js 模拟QQ聊天窗口图片播放效果(带滚轮缩放)

    页面效果如下: 完整代码如下: <!DOCTYPE html> <html> <head> <title>Test</title> < ...

  5. 使用Windows2003创建DHCP服务器 - 进阶者系列 - 学习者系列文章

    Windows 2003提供的DHCP服务还是挺强大的.下面大概介绍下DHCP服务器的配置. 1.  通过控制面板安装DHCP服务 2.  打开DHCP配置项 3.  选择 新建作用域 4.  输入名 ...

  6. 使用rem设计移动端自适应页面二(转载)

    由于日常需求以无线居多,所以可以在业务中做一些尝试,如 rem,刚接触这个特性的时候,曾经一度爱不释手,仿佛在无线开发的坎坷路上寻找到一条捷径.然而随着使用范围的扩大,慢慢的发现了一些使用 rem 带 ...

  7. python进程池剖析(三)

    之前文章对python中进程池的原理.数据流以及应用从代码角度做了简单的剖析,现在让我们回头看看标准库中对进程池的实现都有哪些值得我们学习的地方.我们知道,进程池内部由多个线程互相协作,向客户端提供可 ...

  8. 动画操作 (Applying Animations) ngAnimate12

    动画操作 (Applying Animations) ngAnimate step 12 1.切换目录 git checkout step-12 npm start 2.效果图 这里在点击右边的缩略图 ...

  9. 树莓派学习笔记 1 -- 硬件的需求以及raspbian系统的安装

    树莓派(Raspberry Pi) --  基于Linux系统的大小只有信用卡大小的卡片式机器.  按照发明者的想法,他是想降低学习程序开发的成本而设计制作的这款产品.你可以理解为一个简陋版的电脑.树 ...

  10. 让VS2010记住TFS的登陆用户名和密码

    用VS进行团队开发的都知道,每次打开VS连接TFS的时候,都要提示输入用户名和密码,每次都这样无疑感觉太多此一举了(当然你不想别人操作你的电脑就直接进入项目就没必要这么做),为了像连接远程那样可以记住 ...