Redux实现原理

不同组件需要依赖同一个数据的时候,就需要状态提升至这些组件的根组件。

redux是状态统一管理工具,需要使用它的原因是: 组件之间通信统一管理,方便代码维护。

React中有一个特性context,只要某个组件使用context存储了数据,那么这个组件的所有子组件都可以访问该context内容,并且还可以修改它。就像是这个组件的全局变量,它的所有子组件都可以访问这个全局变量。

如下图,假设要更改主题颜色,那么在Index根组件的context中存储当前主题色,那么它的子组件header,footer,title,menu都可以访问到,并且可以修改。

Redux的统一数据管理,与React的关联之处利用的就是context特性

但是context有一个缺点就是,所有的子组件都可以修改共享的内容,每个组件都能够改 context 里面的内容会导致程序的运行不可预料。所以React团队的做法就是,提高修改的门槛,修改数据时统一调用dispatch函数,并且需要传入修改类型,如果dispatch中不存在该类型,则不允许修改。

接下来简单推理一下redux的实现过程(假设有组件Index,Header,Footer)。

1、第一步是需要有一个存储数据的对象,定义为appState

   	var appState = {
themeColor:'red',
themeBackground:'black'
};
//还需要一个修改数据的函数
function dispatch(action){
switch(action.type){
case 'UPDATE_THEME_COLOR':
appState.themeColor = action.color;break;
case 'UPDATE_THEME_BACKGROUND':
appState.themeBackground = action.color;break;
default break;
}
}
//修改完数据,就需要重新渲染了,定义渲染函数
function renderApp(state){
renderHeader(state);
renderFooter(state);
}
function renderHeader(state){
header = document.getElementById('header');
header.style.color = state.color;
header.style.backgroundColor = state.themeBackground;
}
function renderFooter(state){
title = document.getElementById('footer');
footer.style.color = state.color;
footer.style.backgroundColor = state.themeBackground;
} dispatch({type:'UPDATE_THEME_COLOR',color:'green'})//修改主题色
renderApp(appState)//执行重新渲染

2、封装state和dispatch,命名为store,方便复用,同时添加监听,当数据变化时,通知订阅者重新渲染

	function createStore(state,stateChanger){
const getState = () => state;//获取数据
const listeners = [];
const subscribe = (listener) => listeners.push(listener);
const dispatch = (action) => {
stateChanger(state, action);
listeners.forEach((listener)=>listener());
};
return {getState,dispatch,subscribe};
}
//实例化一个store。为了统一命名,dispatch改为stateChanger,appState改名为state
const store = createStore(state,stateChanger);
renderApp(store.getState());//首次渲染
store.subscribe(renderApp(store.getState()));//添加订阅者,监听到变化就重新渲染数据
store.dispatch({type:'UPDATE_THEME_COLOR',color:'green'});//修改主题

3、第二步有一个很大的缺陷就是,每次修改数据就重新全部渲染一遍,对性能影响很大。

优化点是: 判断数据是否有变化,如果没变化就不需要重新渲染,另外stateChanger与state合为一体。

	//此处用到es6的浅复制,例如
let a = { name:'HAPPY',attr:{age:23},common:{sex:'女'}};
let b = {...a,attr:{age:24}};//b为{name:'HAPPY',attr:{age:24},common:{sex:'女'}}
a.name===b.name//true
a.attr===b.attr//false
a.common===b.common//true
//这样有50%以上的复用率
function stateChanger(state,action){
if(!state){
return {
themeColor:'red',
themeBackground:'black'
}
}
const newState = {...state};
switch(action.type){
case 'UPDATE_THEME_COLOR':
return {
...newState,
themeColor: action.color
};
case 'UPDATE_THEME_BACKGROUND':
return {
...newState,
themeBackground: action.color
};
default
return newState;
}
}
//那么渲染的时候需要知道oldState和newState,这样才能对比数据的变化
function renderApp(state,oldState={}){//此处oldState放在后面,并且给默认值,是为了兼容首次渲染,首次渲染olsState是没有值的,所以给默认值
if(oldState===state){
return;
}
renderHeader(oldState,state);
renderFooter(oldState,state);
}
//如果renderHeder需要的渲染的数据是state内的子对象,那么在renderHeader渲染之前,也需要判断一下数据是否有变化,此处举的例子state结构简单,所以不需要判断。 function createStore(stateChanger){//此处是优化state与stateChanger结合。
let state = null
const getState = () => state;//获取数据
const listeners = [];
const subscribe = (listener) => listeners.push(listener);
const dispatch = (action) => {
state = stateChanger(state, action);
listeners.forEach((listener)=>listener());
};
dispatch({});//初始化,获取state
return {getState,dispatch,subscribe};
}
//最后需要oldState,那么我们就需要定义oldState
const store = createStore(stateStranger);
const oldState = store.getState();
store.subscribe(()=>{
const newState = store.getState();
renderApp(oldState,newState);
oldState = newState;
});//添加监听
renderApp(store.getState())//首次渲染
store.dispatch({type:'UPDATE_THEME_COLOR',color:'green'});//修改主题

4.此时的stateChanger是一个纯函数,就是内部逻辑只与参数有关,并且无副作用(也就是对其他数据没有任何影响)。它要做的仅仅是 —— 初始化和计算新的 state。(并不会修改state,因为我们每次返回的都是新的对象)

而这个函数就是redux中的reducer,那么我们给stateChanger改名字为reducer.现在我们的redux就实现完成了。

	 function reducer(state,action){...}
function createStore(reducer){...}
//接下来我们就可以定义不同的reducer,生成不同的store了,并且修改监听,例如
function themeReducer(state,action){...}
const store = createStore(themeReducer);
// 监听数据变化重新渲染页面
store.subscribe(() => renderApp(store.getState()))
// 首次渲染页面
renderApp(store.getState())
// 后面可以随意 dispatch 了,页面自动更新
store.dispatch(...)
但是我们怎么跟react进行连接呢?怎么把redux用在react中呢?就需要react-redux来连接。下一篇继续

参考教程:(http://huziketang.mangojuice.top/books/react)

redux原理的更多相关文章

  1. 轻松理解Redux原理及工作流程

    轻松理解Redux原理及工作流程 Redux由Dan Abramov在2015年创建的科技术语.是受2014年Facebook的Flux架构以及函数式编程语言Elm启发.很快,Redux因其简单易学体 ...

  2. [转载]Redux原理(一):Store实现分析

    写在前面 写React也有段时间了,一直也是用Redux管理数据流,最近正好有时间分析下源码,一方面希望对Redux有一些理论上的认识:另一方面也学习下框架编程的思维方式. Redux如何管理stat ...

  3. Redux原理(一):Store实现分析

    写在前面 写React也有段时间了,一直也是用Redux管理数据流,最近正好有时间分析下源码,一方面希望对Redux有一些理论上的认识:另一方面也学习下框架编程的思维方式. Redux如何管理stat ...

  4. 解析:让你弄懂redux原理

    作者: HerryLo 本文永久有效链接: https://github.com/AttemptWeb...... Redux是JavaScript状态容器,提供可预测化的状态管理. 在实际开发中,常 ...

  5. Redux 原理和简单实现

    前端开发中React + Redux 是大部分项目的标配,Redux也是我喜欢的库之一,他的源码也拜读过几遍,每次都有很多收获,尤其他的中间件设计模式,对自己封装一些库提供了思想上的指导. Redux ...

  6. react-redux原理分析

    写在前面 之前写了一篇分析Redux中Store实现的文章(详见:Redux原理(一):Store实现分析),突然意识到,其实React与Redux并没有什么直接的联系.Redux作为一个通用模块,主 ...

  7. React + Redux 入坑指南

    Redux 原理 1. 单一数据源 all states ==>Store 随着组件的复杂度上升(包括交互逻辑和业务逻辑),数据来源逐渐混乱,导致组件内部数据调用十分复杂,会产生数据冗余或者混用 ...

  8. react-redux原理

    react-redux原理分析 写在前面 之前写了一篇分析Redux中Store实现的文章(详见:Redux原理(一):Store实现分析),突然意识到,其实React与Redux并没有什么直接的联系 ...

  9. 对redux的理解

     redux原理 某公司有物流(actionType).电商(actionType).广告(actionType)3块业务,在公司财务系统(state)统一记录着三块业务分别赚取到的资金.某天,电商业 ...

随机推荐

  1. associatedtype关联类型

    associatedtype关联类型   定义一个协议时,有的时候声明一个或多个关联类型作为协议定义的一部分将会非常有用.关联类型为协议中的某个类型提供了一个占位名(或者说别名),其代表的实际类型在协 ...

  2. ZBrush带你发掘脸部雕刻的秘诀(上)

    骨骼,是一门基础艺术,几百年来一直为伟大的艺术大师所研究,它曾经,也将一直是创作现实且可信角色的关键,提高骨骼知识更将大大提高雕刻技能. 当然,这对于现实角色很重要,对卡通和风格化的角色也同样重要,底 ...

  3. mysql 最大连接数

    方式一: 一次性修改  服务重启后还原 查看  show variables like 'max_connections%'; 修改 set GLOBAL max_connections=1024; ...

  4. centos7 修改默认语言

    vi /etc/locale.conf # 修改成英文 LANG="en_US.UTF-8" # 修改成中文 LANG="zh_CN.UTF-8"

  5. 计蒜客 阿里天池的新任务—简单( KMP水 )

    链接:传送门 思路:KMP模板题,直接生成 S 串,然后匹配一下 P 串在 S 串出现的次数,注意处理嵌套的情况即可,嵌套的情况即 S = "aaaaaa" ,P = " ...

  6. 【转】H5 input search 提交事件

    欲实现一个文字搜索的功能,要求输入时,键盘回车按钮提示显示为“搜索”.效果如下: 开始~ input type=text并不能达到这种效果,google了一下,HTML5 增加的type=search ...

  7. 在 vue-test-utils 中 mock 全局对象

    vue-test-utils 提供了一种 mock 掉 Vue.prototype 的简单方式,不但对测试用例适用,也可以为所有测试设置默认的 mock. mocks 加载选项 mocks 加载选项 ...

  8. RobotFrameWork+APPIUM实现对安卓APK的自动化测试----第六篇【AppiumLibrary等待函数介绍】

    http://blog.csdn.net/deadgrape/article/details/50622441 废话不多说,少年们请看下面. Wait Until Page Contains text ...

  9. 使用动态代理实现dao接口

    使用动态代理实现dao接口的实现类 MyBatis允许只声明一个dao接口,而无需写dao实现类的方式实现数据库操作.前提是必须保证Mapper文件中的<mapper>标签的namespa ...

  10. git-osc自己定义控件之:CircleImageView

    git-osc自己定义控件之:CircleImageView 一.CircleImageView的使用 在项目中能够发现,用户的头像都是圆形的.感觉非常好奇,昨天最终发现了,原来是自定了一个Image ...