redux异步action_React躬行记(12)——Redux中间件

  Redux的中间件(Middleware)遵循了即插即用的设计思想,出现在Action到达Reducer之前(如图10所示)的位置。中间件是一个固定模式的独立函数,当把多个中间件像管道那样串联在一起时,前一个中间件不但能将其输出传给下一个中间件作为输入,还能中断整条管道。在引入中间件后,既能扩展Redux的功能,也能增强dispatch()函数,适应不同的业务需求,例如通过中间件记录日志、报告奔溃或处理异步请求等。

b6eefb5e54e8e134df71cf873b076608.png
图10 中间件管道

一、开发模式

  在设计中间件函数时,会遵循一个固定的模式,如下代码所示,使用了柯里化、高阶函数等函数式编程中的概念。

function middleware(store) {return function(next) {return function(action) {return next(action);};};
}

  middleware()函数接收一个Store实例,返回值是一个接收next参数的函数。其中next也是一个函数,用来将控制权转移给下一个中间件,从而实现了中间件之间的串联,它会返回一个处理Action对象的函数。由于闭包的作用,在这最内层的函数中,依然能调用外层的对象和函数,例如访问action所携带的数据、执行store中的dispatch()或getState()方法等。示例中的middleware()函数只是单纯的将接收到的action对象转交给后面的中间件,没有对其做额外的处理。

  之所以将中间件函数写成层层嵌套的模式,有以下几个原因。

(1)扩展Redux需要遵循函数式编程的设计思想。

(2)柯里化让中间件更容易处理数据流,即便于形成数据处理管道。

(3)每个中间件的职责单一(即函数代码尽量少),通过嵌套组合完成复杂功能。

  利用ES6中的箭头函数能将middleware()函数改写得更加简洁,如下所示。

const middleware = store => next => action => {return next(action);
};

二、applyMiddleware()

  Redux提供了组织中间件的applyMiddleware()函数,在阅读过此函数的源码(如下所示)之后,才能更好的理解中间件的开发模式。

function applyMiddleware(...middlewares) {return createStore => (...args) => {const store = createStore(...args);let dispatch = () => {throw new Error("Dispatching while constructing your middleware is not allowed. " +"Other middleware would not be applied to this dispatch.");};const middlewareAPI = {getState: store.getState,dispatch: (...args) => dispatch(...args)};const chain = middlewares.map(middleware => middleware(middlewareAPI));dispatch = compose(...chain)(store.dispatch);return { ...store, dispatch };};
}

1)源码分析

  接下来会分析函数中的代码,为了便于理解,在说明中还会给出相应的语句。

  (1)利用剩余参数(...middlewares)的方式,applyMiddleware()函数可以接收任意多个中间件。

  (2)返回一个接收createStore参数的函数,在函数体中调用Redux的createStore()函数,得到一个store实例。

const store = createStore(...args);

  (3)middlewareAPI变量包含了中间件需要的参数,即store.getState()和dispatch()方法。

const middlewareAPI = {getState: store.getState,dispatch: (...args) => dispatch(...args)
};

  (4)将middlewareAPI变量传递给每个中间件并调用一次,再将返回的函数组成一个新数组。

const chain = middlewares.map(middleware => middleware(middlewareAPI));

  (5)通过compose()增强dispatch()方法,compose()是Redux内置的函数,可从右向左合成多个函数,例如compose(f1, f2, f3)(arg)相当于f1(f2(f3(arg)))。

dispatch = compose(...chain)(store.dispatch);

  (6)最终得到一个对象,包含store实例的方法和增强后的dispatch()方法。

return { ...store, dispatch };

2)使用方式

  applyMiddleware()函数有两种使用方式,下面用一个例子来演示,先定义两个中间件:m1和m2,以及一个Reducer函数:caculate()。

const m1 = store => next => action => {console.log("m1");return next(action);
};
const m2 = store => next => action => {console.log("m2");return next(action);
};
function caculate(previousState = {digit:0}, action) {let state = Object.assign({}, previousState);switch (action.type) {case "ADD":state.digit += 1;break;case "MINUS":state.digit -= 1;}return state;
}

  (1)applyMiddleware()是一个三级柯里化的函数,如果要使用,那么可以像下面这样调用,先传两个中间件,再传createStore()函数,最后传caculate()函数。

let store = applyMiddleware(m1, m2)(createStore)(caculate);

  (2)applyMiddleware()函数还可以作为createStore()的最后一个参数,如下代码所示,第一个参数是caculate()函数,第二个参数是applyMiddleware()函数的结果。

let store = createStore(caculate, applyMiddleware(m1, m2));

  在applyMiddleware()函数的内部,chain数组的值是[m1(next), m2(next)],由m1和m2返回的包含next参数的函数组成。增强后的dispatch()方法通过m1(m2(store.dispatch))得到,具体如下所示。

dispatch = action => {console.log("m1");return next(action);
};

  当调用store实例的dispatch()方法(如下代码所示)时,会先输出“m1”;然后调用next()函数,也就是m2(next),输出“m2”;最后调用m2中的next()函数,也就是dispatch()方法。注意,不能在中间件中直接调用dispatch()方法,以免造成死循环。

store.dispatch({ type: "ADD" });

三、redux-thunk

  如果要在Redux中处理异步请求,那么可以借助中间件实现,目前市面上已有很多封装好的中间件可供使用,例如redux-thunk、redux-promise或redux-saga等。本节将着重讲解redux-thunk中间件,其核心代码如下所示。

function createThunkMiddleware(extraArgument) {return ({ dispatch, getState }) => next => action => {if (typeof action === "function") {return action(dispatch, getState, extraArgument);}return next(action);};
}

  首先检测action的类型,如果是函数,那么就直接调用并将dispatch、getState和extraArgument作为参数传入;否则就调用next参数,转移控制权。redux-thunk其实扩展了dispatch()方法,使其参数既可以是JavaScript对象,也可以是函数。

  接下来用一个简单的例子演示redux-thunk的用法,如下代码所示,首先通过import引入redux-thunk,并定义一个异步Action。

import thunk from "redux-thunk";
function asynAction() {return dispatch => {fetch("server.php").then(response => response.json(),error => console.log(error)).then(data => {dispatch(data);        //{type: "ADD"}});};
}

  然后让asynAction()函数返回一个接收dispatch参数的函数,在函数体内通过fetch()函数请求服务端的资源,再利用得到的Promise处理获取到的数据。在第二个then()方法中,data参数被赋为{type: "ADD"},也就是server.php响应的数据,其代码如下所示。

<?php
$json = ['type' => 'ADD'
];
echo json_encode($json);

  再接入redux-thunk中间件,即将thunk传给applyMiddleware()函数,最后发送一个异步Action,如下所示。

let store = createStore(caculate, applyMiddleware(thunk));
store.dispatch(asynAction());

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/356978.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

监控与管理

本文是我们名为“ Spring Integration for EAI ”的学院课程的一部分。 在本课程中&#xff0c;向您介绍了企业应用程序集成模式以及Spring Integration如何解决它们。 接下来&#xff0c;您将深入研究Spring Integration的基础知识&#xff0c;例如通道&#xff0c;转换器和适…

郑州智慧岛大数据管委会_数据科学融通应用数学 ‖ 智慧岛大讲堂

8月6日上午&#xff0c;郑东新区智慧岛大数据实验区管委会&#xff0c;举办了“数据科学融通应用数学”专题智慧岛大讲堂讲座&#xff0c;邀请到北京大学数学科学学院副教授、大数据分析与应用技术国家工程实验室郑州数字创新中心主任卢朓为本次大讲堂活动做主题分享&#xff0…

db2 获取返回的游标_MySQL ------ 存储过程与游标简单使用

存储过程小例子如完成以下事情&#xff0c;获得与之前一样的订单合计&#xff0c;但需要对合计增加营业税&#xff0c;不过只针对某些顾客主要就是&#xff1a;1、获得合计2、把营业税有条件的添加到合计 3、返回合计&#xff08;带或不带税&#xff09;delimiter $$-- 存储过程…

hdu_5761_Rower Bo(xjb猜公式)

题目链接&#xff1a;hdu_5761_Rower Bo 题意&#xff1a; 让你求一个物理问题 题解&#xff1a; xjb猜公式&#xff0c;由题目样例可得&#xff0c;答案为8/7&#xff0c;然后我们可以xjb猜出公式为v1*a/(v1*v1-v2*v2)&#xff0c;然后特判一下a0和v1<v2的情况就OK 1 #incl…

netbeans 源文件_具有NetBeans,嵌入式GlassFish,JPA和MySQL数据源的Arquillian

netbeans 源文件这是一个偶然的帖子。 我一直在研究交易CDI观察者&#xff0c;并尝试使用嵌入式GlassFish对它进行一些集成测试。 但是令人惊讶的是&#xff0c;这种方法不能很好地工作&#xff0c;我仍在弄清楚&#xff0c;使用普通的嵌入式GlassFish时问题出在哪里。 同时&am…

vue 子父组件周期顺序_父组件和子组件生命周期钩子执行顺序是什么?

加载页面执行步骤1、父组件&#xff1a;beforeCreate -> created -> beforeMount2、子组件&#xff1a;beforeCreate -> created -> beforeMount -> mounted3、父组件&#xff1a;mounted销毁组件执行步骤1、父组件&#xff1a;beforeDestroy2、子组件&#xff…

某个元素的距离页面的左边距_在机检测圆心距

点击蓝色标题关注&#xff01;新年伊始&#xff0c;小伙伴们开工了&#xff0c;小编也来给大家加油助威。精密加工过程中&#xff0c;需要严格管控每道工序的加工误差&#xff0c;保证零件最终加工合格。那么该如何快速而又准确地判断&#xff0c;当前工序加工尺寸是否合格&…

复合设计模式示例

本文是我们名为“ Java设计模式 ”的学院课程的一部分。 在本课程中&#xff0c;您将深入研究大量的设计模式&#xff0c;并了解如何在Java中实现和利用它们。 您将了解模式如此重要的原因&#xff0c;并了解何时以及如何应用模式中的每一个。 在这里查看 &#xff01; 目录 …

kafka 不同分区文件存储_大白话 + 13 张图解 Kafka

前言应大部分的小伙伴的要求&#xff0c;在Yarn之前先来一个kafka的小插曲&#xff0c;轻松愉快。一、Kafka基础消息系统的作用应该大部份小伙伴都清楚&#xff0c;用机油装箱举个例子所以消息系统就是如上图我们所说的仓库&#xff0c;能在中间过程作为缓存&#xff0c;并且实…

如何恢复被删短信_手机便签记事本的内容如何恢复?快试试这款好用的便签

手机便签记事本是很多人日常工作和生活中都会用到的一种辅助工具&#xff0c;它不但可以记录我们的待办事项&#xff0c;经常坚持使用的话&#xff0c;还能养成良好的生活节奏。常在河边走哪有不湿鞋&#xff0c;很多人都发生过误删的情况&#xff0c;手机便签记事本的内容如何…

分享一些在内网操作的远程办公经验

前言大家好&#xff0c;我叫孙叫兽&#xff0c;本期内容给大家分享一群在内网操作的程序员远程办公经验。为啥说是内网呢&#xff0c;因为从事的开发项目比较保密&#xff0c;比如某银行总行的新一代智能柜台项目。这些平时开发的内容不能直接和互联网进行连接。只能通过行方的…

OpenMap教程–第1部分

介绍 本系列教程将向您展示如何使用OpenMap GIS Java Swing库构建Java应用程序。 OpenMap的开发人员指南是非常有用的文档&#xff0c;描述了OpenMap的体系结构&#xff0c;但没有说明如何逐步启动和构建应用程序。 源代码附带的示例很有用&#xff0c;但还不够。 OpenMap是用…

killall 后面信号_Linux 下使用 killall 命令终止进程的 8 大用法

Linux 的命令行提供很多命令来杀死进程。比如&#xff0c;你可以向 kill 命传递一个PID来杀死进程&#xff1b;pkill 命令使用一个正则表达式作为输入&#xff0c;所以和该模式匹配的进程都被杀死。但是还有一个命令叫 killall &#xff0c;默认情况下&#xff0c;它精确地匹配…