redux之applyMiddleware
redux之所以伟大就在于中间件了,中间件为redux提供了无限可能。redux中中间件是一个不太容易理解的概念,因为涉及到compose、hoc等函数式的概念,看源代码总是懵懵的感觉。今天我们就来详细解剖一下伟大的applyMiddleware吧。
applyMiddleware只有短短三十多行,可见作者功力。先简单说下中间件是啥,在redux中,当你要dispatch一条命令给reducer时,预先定义的中间件会对这条命令进行各种转换,有的中间件会记录这个过程,有的会对payload进行转换,甚至可以终止这次dispatch(短路),再发起另一个命令。下面我们切入正题。
入参
export default applyMiddleware(...middlewares) {
...
}
首先通过rest参数的方式获取一个中间件数组。
返回值
export default applyMiddleware(...middlewares) {
return (createStore) => (reducer, preloadedState, enhancer) => {
return {
...store,
dispatch
}
}
}
可以看到返回值是一个HOC,它接受一个createStore函数,因此你还可以使用applyMiddleware来创建store
const finalCreateStore = applyMiddleware(thunk, log)(createStore)
const store = finalCreateStore(reducer, preloadedState, enhancer)
重点来了
先把代码贴上,
const store = createStore(reducer, preloadedState, enhancer)
let dispatch = store.dispatch
let chain = []
const middlewareAPI = {
getState: store.getState,
dispatch: (action) => dispatch(action)
}
chain = middlewares.map(middleware => middleware(middlewareAPI))
dispatch = compose(...chain)(store.dispatch)
这里不对compose做过多解释了,请大家自行了解。
我们直接锁定到chain = middlewares.map(middleware => middleware(middlewareAPI)),这里每个middleware接受middlewareAPI作为参数,那么我们知道了middleware大概的样子即,
const middleware = function({getState : function, dispatch: function}) {
...
}
我们继续,下一行代码就是精髓了,dispatch = compose(...chain)(store.dispatch),首先compose(...chain)不难理解,即从右到左chain中的每个middleware(注意这里的middleware并不是一开始传进来的middleware了,是一个函数执行的结果)所执行的结果将作为左边middleware的入参,后面以此类推,那么store.dispatch理所当然成为chain中最后一个middleware的入参,那么我们知道最后一个middleware应该是这样的。
const middleware = function({getState : function, dispatch: function}) {
return (store.dispatch) {
...
}
}
这里我们换一个更加普适的写法,
const middleware = function({getState : function, dispatch: function}) {
return (next) => {
...
}
}
难点来了,到这里显而易见的线索就不多了,由代码可以看出,最终这个compose后的dispatch被return出去了,我们在实际开发中,比如dispatch({type: 'INCREMENT'}),这里的dispatch并不是store.dispatch(记为realDispatch),而是return的那个dispatch(我们记为fakeDispatch),因此又多了不少线索,我们fakeDispatch的action最终要传给realDispatch来触发reducer中的逻辑,那么我们如何传递action呢? 我们回到我们现在猜测的middleware代码中,最后一个middleware的next就是realDispatch,现在我们把注意力放在这里,即应该要给左边的middleware返回什么呢,
// 这是middleware的执行结果
const middleare = (next) => {
// 这里要返回什么呢
}
如果我们返回一个非函数,那么这个middleware以后就没啥事了,显然不行,因为我还需要接受action,再调用realDispatch;所以我们需要返回一个闭包函数,这个函数保留了next的访问引用,那么我们继续猜测最后一个middleware,
const middleware = (next) => {
return (action) => {
next(action)
}
}
OK,现在我们可以保证其可以将action顺利送给reducer了,回顾下compose,我们可以看到,(action) => {//...} 就是每个函数的返回值,它会作为入参传给左侧函数,这个入参即next,这样action从fakeDispatch出发,经过这样一个个action作为参数的next函数,最终到达realDispatch这个next并触发reducer执行。下面我们来个完整的middleware,
const middleware = (getState, dispatch) => (next) => (action) => { // 具体逻辑...}
如果你对redux-logger和redux-thunk的源码有些记忆的话你会很熟悉这个函数签名了吧。原来这个难懂的HOC就是这么来的。这你也就知道了redux-thunk为何要放到最左边,因为它要能够短路,即action从fakeDispatch传进来时,它是第一个接收的,这样它就可以根据anction类型来确定是否终止这个dispatch,下面我们写下简版redux-logger和redux-thunk,实战下。
redux-thunk
const middleware = (getState, dispatch) => (next) => (action) => {
if(typeof action === 'function') {
action(getState, dispatch) // 短路
} else {
next(action)
}
}
redux-logger
const middleware = (getState, dispatch) => (next) => (action) => {
console.log('开始触发reducer')
next(action)
console.log('结束触发reducer')
}
redux之applyMiddleware的更多相关文章
- redux 中间件 --- applyMiddleware 源码解析 + 中间件的实战
前传 中间件的由来 redux的操作的过程,用户操作的时候,我们通过dispatch分发一个action,纯函数reducer检测到该操作,并根据action的type属性,进行相应的运算,返回st ...
- react+redux教程(七)自定义redux中间件
今天,我们要讲解的是自定义redux中间件这个知识点.本节内容非常抽象,特别是中间件的定义原理,那多层的函数嵌套和串联,需要极强逻辑思维能力才能完全消化吸收.不过我会多罗嗦几句,所以不用担心. 例子 ...
- redux middleware 的理解
前言 这几天看了redux middleware的运用与实现原理,写了一个百度搜索的demo,实现了类似redux-thunk和redux-logger中间件的功能. 项目地址:https://git ...
- Flux --> Redux --> Redux React 入门
本文的目的很简单,介绍Redux相关概念用法 及其在React项目中的基本使用 假设你会一些ES6.会一些React.有看过Redux相关的文章,这篇入门小文应该能帮助你理一下相关的知识 一般来说,推 ...
- Redux学习(2) ----- 异步和中间件
Redux中间件,其实就是一个函数, 当我们发送一个action的时候,先经过它,我们就可以对action进行处理,然后再发送action到达reducer, 改变状态,这时我们就可以在中间件中,对a ...
- redux源码解析-函数式编程
提到redux,会想到函数式编程.什么是函数式编程?是一种很奇妙的函数式的编程方法.你会感觉函数式编程这么简单,但是用起来却很方便很神奇. 在<functional javascript> ...
- redux的源码解析
一. redux出现的动机 1. Javascript 需要管理比任何时候都要多的state2. state 在什么时候,由于什么原因,如何变化已然不受控制.3. 来自前端开发领域的新需求4. 我们总 ...
- Flux --> Redux --> Redux React 基础实例教程
本文的目的很简单,介绍Redux相关概念用法 及其在React项目中的基本使用 假设你会一些ES6.会一些React.有看过Redux相关的文章,这篇入门小文应该能帮助你理一下相关的知识 一般来说,推 ...
- 再探Redux Middleware
前言 在初步了解Redux中间件演变过程之后,继续研究Redux如何将中间件结合.上次将中间件与redux硬结合在一起确实有些难看,现在就一起看看Redux如何加持中间件. 中间件执行过程 希望借助图 ...
随机推荐
- python的apidoc使用
一.apidoc的安装 npm install apidoc -g -g参数表示全局安装,这样在哪儿都能使用. 二.apidoc在python接口代码中的使用 def index(): "& ...
- 开源虚拟化KVM(一)搭建部署与概述
一,KVM概述 1.1 虚拟化概述 在计算机技术中,虚拟化意味着创建设备或资源的虚拟版本,如服务器,存储设备,网络或者操作系统等等 [x] 虚拟化技术分类: 系统虚拟化(我们主要讨论的反向) 存储虚拟 ...
- jdbc访问pipelinedb
建立Stream及视图 pipeline.execute("create stream caesar(name text,info json);") #创建stream,字段nam ...
- Systemd 教程
目录 Systemd 教程 sshd.service配置模板 开机启动 启动服务 停止服务 配置文件 [Unit] 区块:启动顺序与依赖关系 [Service] 区块:启动行为 1.启动命令 2.启动 ...
- php+javascript实现的动态显示服务器运行程序进度条功能示例
本文实例讲述了php+javascript实现的动态显示服务器运行程序进度条功能.分享给大家供大家参考,具体如下: 经常有这样的业务要处理,服务器上有较多的业务需要处理,需要分批操作,于是就需要一个提 ...
- 重启虚拟机后dhclient进程未运行解决办法
问题分析 重启虚拟机后,dhclient进程未运行的根因通常为: 1.NetworkManager未开启自启动导致的dhclient进程未运行 2.网卡设置未纳入NetworkManager管理导致的 ...
- FFmpeg 开发环境搭建及第一个程序 Hello FFmpeg 编写
1. FFmpeg 的安装 ./configure make make install 默认会将 FFmpeg 安装至 /usr/local 目录下(可通过 configure 使用 "-p ...
- Hillstone设备管理-恢复出厂设置
1.CLI命令行操作 unset all: 根据提示选择是否保存当前配置y/n: 选择是否重启y/n: 系统重启后即恢复到出厂设置. 2.webUI操作 “系统”—“配置”,点击“清除”按钮,系统会提 ...
- layer弹出层不居中解决方案(转)
@感谢参考文章 原文内容: 一.问题描述 用layer做操作结果提示时,发现如果页面超出屏幕的高度时,弹出的提示不是屏幕居中,而是在页面高度的中间,如果一个页面的高度比较大,就看不到提示了. 还有一种 ...
- 解决layui table方法渲染时时间格式问题
在显示时间时没有成功 ,{field:'showTime',title:'要显示的时间'} 崎岖过程就不详述了,直接上干货 @官网相关文档1.@官网相关文档2.@参考文章1.@参考文章2 浏览了很多资 ...