Redux----Regular的Redux实现整理
Regular的Redux实现整理
什么问题?
组件的树形结构决定了数据的流向,导致的数据传递黑洞
怎么解决?
所有组件都通过中介者传递共享数据
方案:
中介者:
(function createStore() {
var store;
return function() {
if(!store) {
store = new Regular;
}
return store;
}
})()
组件A修改数据
define(['./store.js'], function(createStore) {
var A = Regular.extend({
name: "组件A",
data: {
title: '标题'
},
getData: function() {
this.data.title = createStore().data.title;
},
setData: function() {
store.data.title = '新标题'
//通知所有其他组件
store.$emit('change', {
title: '新标题'
})
}
});
return A;
});
其他组件可以监听,也可以主动拿:
define(['./store.js'], function(store) {
var B = Regular.extend({
name: "组件B",
init: function() {
createStore().$on('change', function(newTitle){
this.data.title = newTitle
})
}
});
return B;
});
两个问题:
1 store.data可以直接被访问和修改->data和store分开&通过接口拿
2 只需要订阅和派发(派发的时候会把数据存起来并通知其他订阅者)
(function createStore() {
var store;
return function() {
if(!store) {
var store = new Regular;
var state = {};
store.getState = function(){ return state; };
store.subscribe = function(listener) { store.$on('change', listener); }
store.dispatch = function(action) {
if(action.type == 'changeTitle') {
state.title = action.data.title;
}
store.$emit('change', state);
}
}
return store;
}
})()
define(['./store.js'], function(createStore) {
var A = Regular.extend({
name: "组件A",
data: {
title: '标题'
},
getData: function() {
this.data.title = createStore().data.title;
},
setData: function() {
store.dispatch({
type: 'changeTitle',
data: {title: '新标题'}
})
});
return A;
});
define(['./store.js'], function(store) {
var B = Regular.extend({
name: "组件B",
init: function() {
createStore().subscribe(mapState)
},
mapState: function(state) {
this.data.title = state.title
}
});
return B;
});
这个就是基本的redux雏形。后面的其实都是一些改进。
改进1: 数据处理耦合在store当中->抽出reducer并能传入初始state
(function createStore(reducer, initState) {
var store;
return function() {
if(!store) {
var store = new Regular;
var state = initState;
store.getState = function(){ return state; };
store.subscribe = function(listener) { store.$on('change', listener); }
store.dispatch = function(action) {
state = reducer(state, action);
store.$emit('change', state);
}
}
return store;
}
})()
reducer长这样:
function reducer1(state, action) {
switch(action.type) {
case 'CHANGE_TITLE':
//es6写法
//https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators/Spread_operator
//return {
//...state,
//title: action.data.title
//}
return Object.assign({}, state, {title: action.data.title});
break;
}
}
reducer说是一个纯函数,本质上就是它不改变输入的参数,最后返回了一个新对象(规约)。也解决了可变数据结构的问题。
改进2:每个组件都需要依赖store,并且需要调用store的subscribe和dispatch方法->创建顶层容器并且扩展它的组件的能力
const App = Regular.extend({
name: 'App',
template: `
<StoreProvider>
<A />
<B />
</StoreProvider>
`
});
var StoreProvider = Regular.extend({
template: '{#include this.$body}',
config: function(){
this.store = createStore();
},
modifyBodyComponent: function( component ){
component.dispatch = this.store.dispatch.bind(store)
//把订阅这个工作给做了
this.store.subscribe(function () {
var state = this.store.getState();
component.mapState(state);
}.bind(this));
component.getState = this.store.getState.bind(store)
}
});
然后子组件内就可以这样调用了:
define([], function() {
var B = Regular.extend({
name: "组件B",
init: function() {
this.subscribe(mapState)
},
mapState: function(state) {
this.data.title = state.title
}
});
return B;
});
改进3: 这里的store没法传reducer和初始状态,因为你是里面调用的->createStore在外面做,然后把store传给顶层容器
const App = Regular.extend({
name: 'App',
template: `
<StoreProvider store={store}>
<A />
<B />
</StoreProvider>
`
config(data) {
data.store = createStore(reducers, { title: "标题" })
}
});
var StoreProvider = Regular.extend({
template: '{#include this.$body}',
config: function(data){
this.store = data.store;
},
modifyBodyComponent: function( component ){
component.dispatch = this.store.dispatch.bind(store)
//把订阅这个工作给做了
this.store.subscribe(function () {
var state = this.store.getState();
component.mapState(state);
}.bind(this));
//这个方法其实不用了。因为数据通过connect获得了,而初始数据通过一开始就传到store进去了
component.getState = this.store.getState.bind(store)
}
});
改进3: 并且这样每个组件都有一个mapState方法,而且做得事都比较类似就是把state的数据过来-> 抽出一个connect函数
function connect(config, B) {
B.implement({
mapState: function(state) {//this指向B
const mappedData = config.mapState.call(this, state);
mappedData && Object.assign(this.data, mappedData);
}
});
}
connect({
//es6简写
//mapState(state) {
//...
//}
mapState: function(state) {
return {
title: state.title
}
}, A);
以上,就是redux核心的基本实现原理。
中间件是干嘛用的?
https://guoyongfeng.github.io/book/15/04-redux-logger%E7%9A%84%E5%BA%94%E7%94%A8%E5%92%8C%E7%90%86%E8%A7%A3.html
比如上面的logger中间件,就是想在store的dipatch方法里面做点其他事,比如打印下个性化的日志。
怎么实现?
正常想法:代理dispatch方法
function applyMiddleware() {
let store = createStore(reducer1, initState);
store.dispatch = function(action) {
console.log("....")
store.dispatch(action);
console.log("....")
}
return store;
}
后面呢也是改进。。
改进1:不希望改变原先store&dispatch内打印日志部分希望能做其他事情->传递一个middleware回调进去
function logger(dispatch, action) {
console.log("....")
dispatch(action);
console.log("....")
}
function applyMiddleware(middleware) {
let store = createStore(reducer1, initState);
let dispatch = function(action) {
middleware(store.dispatch, action)
}
return Object.assign({}, store, {dispatch: dispatch});
}
var store = applyMiddleware(logger)
改进2: 如何处理多个中间件(每个中间件做的事不一样)而且store的dispatch应该只被执行一次
var store = applyMiddlewares(logger, someMiddleware)
function applyMiddlewares(logger, someMiddleware) {
let store = createStore(reducer1, initState);
let dispatch = function(action) {
//希望一层层代理地执行中间件,最左边的先执行
someMiddleware(logger(store.dispatch, action), action)
}
return Object.assign({}, store, {dispatch: dispatch});
}
//所以logger必须return一个函数,给其他middleware执行
function logger(dispatch) {
return function(action) {
console.log("....")
dispatch(action);
console.log("....")
}
}
改进3: 仔细看这里面的实现发现就是数组的reduceRight方法
function applyMiddlewares(middlewares) {
let store = createStore(reducer1, initState);
let dispatch = middlewares.reduceRight(function(dispatch, middeware) {
return middeware(dispatch);
}, store.dispatch)
return Object.assign({}, store, {dispatch: dispatch});
}
改进4: 中间件里面的next函数是干嘛用的?->其实就是传进去的dispatch方法
function logger(next) {
return function(action) {
console.log("....")
next(action);
console.log("....")
}
}
以上,就是redux中间件实现的基本原理。
Action Creator是什么?
是对dispatch函数参数(也就是action)的一种抽象,便于Action的复用
比如我们这么写:
this.$dispatch({
action: 'CHANGE_TITLE',
data: { title: '新标题' }
})
可能其他组件也需要写相似的代码,你需要复制代码。其实我们可以抽出一个creator。
//这部分可以被复用
const CHANGE_TITLE = 'CHANGE_TITLE';
function changeTitle(newTitle) {
return {
type: CHANGE_TITLE ,
data: {
title: newTitle
}
}
}
this.$dispatch(changeTitle('新标题'))
最后的一张图总结:
作者知乎/公众号:前端疯
Redux----Regular的Redux实现整理的更多相关文章
- 读redux有感: redux原来是这样操作的。
2017.9.10日 教师节 : ~当一个事物你没有接触,但是生活中 常常用到他,你就不得不去了解他了. 注:新手可以看一下,毕竟博主也是个菜鸟,没法写高深的东西,不想看博主扯淡的直接看第三节啦~~ ...
- react第十八单元(redux中间件redux-thunk,redux工程目录的样板代码,规范目录结构)
第十八单元(redux中间件redux-thunk,redux工程目录的样板代码,规范目录结构) #课程目标 中间件:中间件增强redux的可扩展性,实现功能复用的目的. redux-thunk异步逻 ...
- 学习Redux之分析Redux核心代码分析
1. React,Flux简单介绍 学习React我们知道,React自带View和Controller库,因此,实现过程中不需要其他任何库,也可以独立开发应用.但是,随着应用程序规模的增大,其需要控 ...
- Redux 学习(1) ----- Redux介绍
Redux 有三个基本的原则: 1,单一状态树,redux 只使用一个javascript 对象来保存整个应用的状态. 状态树样式如下: const state = { count: 0 } 2,状态 ...
- redux sample with redux source code
code sample没有package.json文件,也就没有任何外部依赖,直接使用redux source code. nodejs对es6的import export还不支持,这里使用了stac ...
- Redux进阶(Redux背后的Flux)
简介 Flux是一种搭建WEB客户端的应用架构,更像是一种模式而不是一个框架. 特点 单向数据流 与MVC的比较 1.传统的MVC如下所示(是一个双向数据流模型) 用户触发事件 View通知Contr ...
- 迷你redux实现,redux是如何进行实现的?
export function createStore(reducer){ let currentState={} let currentListeners=[] function getState( ...
- redux 个人整理
序 本人工作之余的闲暇时间还是很充裕的.在完成经理安排的任务后,基本上都是在自学,主要是阅读各种技术文档.浏览博客.运行别人写的一些前端demo并观赏与赞叹.在ScorpionJay 同学的带领下,我 ...
- Redux教程1:环境搭建,初写Redux
如果将React比喻成士兵的话,你的程序还需要一位将军,去管理士兵(的状态),而Redux恰好是一位好将军,简单高效: 相比起React的学习曲线,Redux的稍微平坦一些:本系列教程,将以" ...
随机推荐
- Java中的表达式运算
1.问题背景 以下代码运行的结果为: A.a的值:8 b的值:7 B.a的值:8 b的值:8 C.a的值:9 b的值:7 D.a的值 ...
- windows驱动程序wdf--KMDF大致框架
继WDM后微软出了WDF,封装了WDM中的一些基本代码逻辑.本人菜鸟,也不知道本质上有何区别,只觉得是多了Wdf开头的函数,基本的编程框架上有点出入. KMDF是WDF的内核级部分,为了理清KMDF的 ...
- jquery.lazyload.js实现图片懒加载
个人理解:将需要延迟加载的图片的src属性全部设置为一张相同尽可能小(目的是尽可能的少占宽带,节省流量,由于缓存机制,当浏览器加载了一张图片之后,相同的图片就会在缓存中拿,不会重新到服务器上拿)的图片 ...
- 当发现你的OpenStack虚拟机网络有问题,不妨先试一下这16个步骤
1. Security Group全部打开,这是最基本的,但是很多人容易忘记 其实遇到过无数这种场景了,Debug了半天网络问题,各种手段都用上了,最后发现安全组竟然没有打开. 2. 通过界面查看虚拟 ...
- 【BZOJ3130】费用流(最大流,二分)
[BZOJ3130]费用流(最大流,二分) 题面 Description Alice和Bob在图论课程上学习了最大流和最小费用最大流的相关知识. 最大流问题:给定一张有向图表示运输网络,一个源点S和一 ...
- 【BZOJ3436】小K的农场(差分约束)
[BZOJ3436]小K的农场(差分约束) 题面 由于BZOJ巨慢无比,使用洛谷美滋滋 题解 傻逼差分约束题, 您要是不知道什么是差分约束 您就可以按下\(Ctrl+W\)了 #include< ...
- Windows Developer Day - Adaptive Cards
概述 Windows Developer Day 在 Modern Application Experience 环节展示了一种可以让开发者以更通用和统一的方式来对卡片对展示和交互的方式,那就是:Ad ...
- Zabbix JMX之tomcat监控
工作原理: 1.JAVA-GATEWAY Zabbix本身不支持直接监控Java,在zabbix 1.8以前,只能使用Zapcat来做代理监控,而且要修改源代码,非常麻烦.所有后来为了解决这个监控问 ...
- golang goroutine的调度
golang goroutine的调度 1.什么是协程? 协程是一种用户态的轻量级线程. 2.进程.线程.协程的关系和区别: * 进程拥有自己独立的堆和栈,既不共享堆,亦不共享栈,进程由操作系统调度. ...
- Xamarin改变移动开发的五个理由
企业开发者不能简单的抛弃现有的桌面和Web应用,然而又不得不忙着创建各种各样的应用,没有太多的预算来开发移动版本,尤其是原生版本. 采用Xamarin,C#开发人员可以使用一份基础代码创建桌面版和移动 ...