redux深入理解之中间件(middleware)
本文代码请看本人github,https://github.com/Rynxiao/redux-middleware
关于redux运用,请看之前一篇文章http://blog.csdn.net/yuzhongzi81/article/details/51880577
理解reduce函数
reduce() 方法接收一个函数作为累加器(accumulator),数组中的每个值(从左到右)开始缩减,最终为一个值。
arr.reduce([callback, initialValue])
关于reduce的用法,这里不再做多述,可以去这里查看
看如下例子:
let arr = [1, 2, 3, 4, 5];
// 10代表初始值,p代表每一次的累加值,在第一次为10
// 如果不存在初始值,那么p第一次值为1
// 此时累加的结果为15
let sum = arr.reduce((p, c) => p + c, 10); // 25
// 转成es5的写法即为:
var sum = arr.reduce(function(p, c) {
console.log(p);
return p + c;
}, 10);
下面我们再来看一个reduce的高级扩展。现在有这么一个数据结构,如下:
let school = {
name: 'Hope middle school',
created: '2001',
classes: [
{
name: '三年二班',
teachers: [
{ name: '张二蛋', age: 26, sex: '男', actor: '班主任' },
{ name: '王小妞', age: 23, sex: '女', actor: '英语老师' }
]
},
{
name: '明星班',
teachers: [
{ name: '欧阳娜娜', age: 29, sex: '女', actor: '班主任' },
{ name: '李易峰', age: 28, sex: '男', actor: '体育老师' },
{ name: '杨幂', age: 111, sex: '女', actor: '艺术老师' }
]
}
]
};
比如我想取到这个学校的第一个班级的第一个老师的名字,可能你会这样写:
school.classes[0].teachers[0].name
这样不就行了么?so easy!是哦,这样写"毫无问题",这个毫无问题的前提是你已经知道了这个值确实存在,那么如果你不知道呢?或许你要这么写:
school.classes &&
school.classes[0] &&
school.classes[0].teachers &&
school.classes[0].teachers[0] &&
school.classes[0].teachers[0].name
我去,好大一坨,不过要在深层的对象中取值的场景在工作中真真实实存在呀?怎么办?逛知乎逛到一个大神的解决方案,如下:
const get = (p, o) => p.reduce((xs, x) => (xs && xs[x] ? xs[x] : null), o);
// call
get('classes', 0, 'teachers', 0, 'name', school); // 张二蛋
是不是很简单,用reduce这个方法优雅地解决了这个问题。
理解redux的compose函数
讲了这么久的reduce,这不是讲redux么?这就尴尬了,下面我们就来看看为什么要讲这个reduce函数。去github上找到redux源码,会看到一个compose.js文件,带上注释共22行,其中就用到了reduce这个函数,那么这个函数是用来做啥的?可以看一看:
export default function compose(...funcs) {
if (funcs.length === 0) {
return arg => arg
}
if (funcs.length === 1) {
return funcs[0]
}
return funcs.reduce((a, b) => (...args) => a(b(...args)))
}
初步看上去貌似就是函数的嵌套调用。我们去搜一下,看哪个地方会用到这个函数,在源码中找一下,发现在applyMiddleware.js中发现了这样的调用:
export default function applyMiddleware(...middlewares) {
return (createStore) => (reducer, preloadedState, enhancer) => {
const store = createStore(reducer, preloadedState, enhancer)
let dispatch = store.dispatch
let chain = []
const middlewareAPI = {
getState: store.getState,
dispatch: (...args) => dispatch(...args)
}
chain = middlewares.map(middleware => middleware(middlewareAPI))
dispatch = compose(...chain)(store.dispatch)
return {
...store,
dispatch
}
}
}
看到熟悉的东西了么?applyMiddleware哟,我们在写中间件必须要用的函数。我们来看一下一个简单的middleware是怎样写的?比如我要写一个loggerMiddleware,那么就像这样:
const logger = store => next => action => {
console.log('action', action);
let result = next(action);
console.log('logger after atate', store.getState());
return result;
}
当我们创建了一个store的时候,我们是这样调用的:
let middlewares = [loggerMiddleware, thunkMiddleware, ...others];
let store = applyMiddleware(middlewares)(createStore)(reducer, initialState);
那么传给compose的funcs实际上就是包含这样的函数的一个数组:
function(next) {
return function(action) {
return next(action);
}
}
当把这样的一个数组传给compose会发生什么样的化学反应呢?稍微看一下应该不难看出,最终会返回一个函数,这个函数是通过了层层middleware的加工,最终的形态仍如上面的这个样子。注意,此时的next(action)并未执行,当执行了
compose(...chain)(store.dispatch)
之后,返回的样子是这样的:
function(action) {
return next(action);
}
各位看官们,看出了一点点什么东西了么?好像createStore中的dispatch呀,没错,这其实也是一个dispatch,只是这个dispatch正一触即发,再等待一个机会。我们有这么一个数量加1的action,类似这样的:
export function addCount() {
return {
type : ADD_COUNT
}
}
// 下面我们来触发一下
dispatch(addCount());
没错,此时的dispatch执行啦,最外层的dispatch执行了会发生什么样的反应呢?看下面:
return next(action);
// 这个next就是dispatch函数,只不过这个dispatch函数在每次执行的时候,会保留
// 上一个middleware传递的dispatch函数的引用,因此会一直的传递下去,
// 直到最终的store.dispatch执行
那么我们去createStore中去看看dispatch函数的定义:
function dispatch(action) {
// ...
try {
isDispatching = true
currentState = currentReducer(currentState, action)
} finally {
isDispatching = false
}
// ...
return action
}
找到这一句
currentState = currentReducer(currentState, action);
当执行了这一步的时候,这一刻,原本传递过来的initialState值已经改变了,那么就会层层执行middleware之后的操作,还记得我们在middleware中这样写了么:
const logger = store => next => action => {
console.log('action', action);
let result = next(action);
console.log('logger after atate', store.getState());
return result;
}
这就是为什么我们会在next执行之后,会取到store中的state的原因。
异步的middlewares
异步的action写法上可能会和立即执行的action不一样,例如是这样的:
// 定义的非纯函数,提供异步请求支持
// 需要在sotre中使用thunkMiddleware
export function refresh() {
return dispatch => {
dispatch(refreshStart());
return fetch(`src/mock/fetch-data-mock.json`)
.then(response => response.json())
.then(json => {
setTimeout(() => {
dispatch(refreshSuccess(json && json.data.list));
}, 3000);
});
}
}
为什么要使用thunkMiddleware呢,我们去找一找thunkMiddleware中到底写了什么?
function createThunkMiddleware(extraArgument) {
return ({ dispatch, getState }) => next => action => {
if (typeof action === 'function') {
return action(dispatch, getState, extraArgument);
}
return next(action);
};
}
const thunk = createThunkMiddleware();
thunk.withExtraArgument = createThunkMiddleware;
export default thunk;
短短14行代码,看这一句:
if (typeof action === 'function') {
return action(dispatch, getState, extraArgument);
}
如果action的类型为function的话,那么就直接执行啦,实际上就是将一个异步的操作转化成了两个立即执行的action,只是需要在异步前和异步后分别发送状态。为什么要分解呢?如果不分解会是什么样的情况?还记得这一行代码吗?
currentReducer(currentState, action);
这里的reducer只接受纯函数,只接受纯函数,只接受纯函数,重要的事情说三遍。所以你传个非纯函数是个什么鬼?那不是直接走switch的default了么?所以得到的state依旧是之前的state,没有任何改变。
redux深入理解之中间件(middleware)的更多相关文章
- redux 中的 redux-thunk(中间件)
前言 空闲时间把redux中的redux-thunk中间件回顾下,因为以前没有写博客的习惯,都怪自己太年轻,好了 其实: redux的核心概念其实很简单:将需要修改的state都存入到sto ...
- 初探中间件(middleware)
初探中间件(middleware) 因为考虑到文章的长度, 所以 BaseHandler 的展开被推迟了. 在 BaseHandler 中隐藏着中间件的信息, 较常见的 SessionMiddlewa ...
- C# Owin 创建与测试自己的中间件Middleware(二)
本文纯属介绍怎么简单地创建自己的Owin.还没有理解owin概念的请看上一篇文章:http://www.cnblogs.com/alunchen/p/7049307.html 目录 1.创建项目 2. ...
- 跟我一起学.NetCore之中间件(Middleware)简介和解析请求管道构建
前言 中间件(Middleware)对于Asp.NetCore项目来说,不能说重要,而是不能缺少,因为Asp.NetCore的请求管道就是通过一系列的中间件组成的:在服务器接收到请求之后,请求会经过请 ...
- ASP.NET Core 开发-中间件(Middleware)
ASP.NET Core开发,开发并使用中间件(Middleware). 中间件是被组装成一个应用程序管道来处理请求和响应的软件组件. 每个组件选择是否传递给管道中的下一个组件的请求,并能之前和下一组 ...
- ASP.NET Core中间件(Middleware)实现WCF SOAP服务端解析
ASP.NET Core中间件(Middleware)进阶学习实现SOAP 解析. 本篇将介绍实现ASP.NET Core SOAP服务端解析,而不是ASP.NET Core整个WCF host. 因 ...
- laravel中间件-----------middleware
middleware中间件 是访问到达服务器后在被对应的路由处理之前所经过的一层过滤层,故称中间件. 中间件是存放在app\http\middleware中,需要定一个 handle 处理方法,在ha ...
- 二、中间件(middleware)
1. 中间件(middleware) Django中的中间件主要实现一些附加功能,在request被用户handler处理前,以及用户handler处理后生存的response进行处理.因此 ...
- 中间件(Middleware)
中间件(Middleware) ASP.NET Core开发,开发并使用中间件(Middleware). 中间件是被组装成一个应用程序管道来处理请求和响应的软件组件. 每个组件选择是否传递给管道中的下 ...
随机推荐
- [图形学] 习题8.12 NLN二维线段裁剪算法实现
Nicholl-Lee-Nicholl二维线段裁剪算法相对于Cohen-Sutherland和Liang-Barsky算法来说,在求交点之前进行了线段端点相对于几个区域的判断,可以确切的知道要求交点的 ...
- rownumber和rowid伪劣用法
select rownum ,deptno,dname loc from dept; select deptno,dname,loc from dept where rownum=1; select ...
- 一次C++调试记录
之前开发用Linux C比较多,C++中的STL 容器基本没有接触过.最近在学习C++,平时用到c++ 17中的部分新特性,下面就简单分享下自己C++的学习流程. 一.环境搭建 本 ...
- Android6.0 中appcompat_v7 报错
更新了AndroidSDK以后 各种错误,新建一个项目会附赠一个appcompat_v7,你只要知道这个是一个兼容包就可以了,具体的特性可以看相关介绍,其实也没啥特别的就是为了兼容低版本的呗, 但是呢 ...
- Servlet转发到JSP页面的路径问题
一.现象与概念 1. 问题 在Servlet转发到JSP页面时,此时浏览器地址栏上显示的是Servlet的路径,而若JSP页面的超链接还是相对于该JSP页面的地址且该Servlet和该JSP页面不在同 ...
- Android之ListView优化
关于ListView几个方面的优化: ListView的大小设定固定值; 复用convertView, 使用ViewHolder提高在容器中查找组件的效率; 使用分页加载; 快速滚动时, item不显 ...
- Python怎么样入门?Python基础入门教程
给大家整理的这套python学习路线图,按照此教程一步步的学习来,肯定会对python有更深刻的认识.或许可以喜欢上python这个易学,精简,开源的语言.此套教程,不但有视频教程,还有源码分享,让大 ...
- CentOS的改变系统启动级别
CentOS7改变系统启动级别 systemctl命令: 文本模式:systemctl set-default multi-user.target 图形模式:systemctl set-def ...
- Java Swing intro
Java Swing intro 如果有Android app开发经验,快速上手Swing不是问题.UI方面有相似的地方. 简单的几行代码就能抛出一个框框,记录一下操作过程 1.先显示一个框框 Era ...
- FreeRTOS——任务管理
1. FreeRTOS 任务不允许以任何方式从实现函数中返回——他们绝不能有一条“return”语句,也不可能执行到函数的末尾.如果一个函数不需要,可以将其删除,如在任务中使用函数vTaskDelet ...