Redux记录:Store是如何自动调用reducers来处理action的

2024-05-14 03:48

本文主要是介绍Redux记录:Store是如何自动调用reducers来处理action的,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Redux记录:Store是如何自动调用reducers来处理action的

作为一个后端程序员,经常也要写一点前端、维护一下前端。因此一直在与前端打交道,但是一直没有理解当用户操作view通过dispatch发出 action之后,我们定义的一系列的reducer是如何来自动执行处理的。

先说结论:当用户操作view之后发出一个action,store会遍历所有的reducers来依次处理这个action来改变state。

今天浏览自己所在公司的官方博客,发现了这篇文章:Redux从设计到源码,仔细拜读了一下,收获很大,也解决了自己一直以来的困惑。

借此机会,自己也梳理一下。

在前端代码,自己经常看到类似如下的代码:

    let store = createStore(reducers, undefined, compose(applyMiddleware(thunk,fetchMiddleware),window.devToolsExtension ? window.devToolsExtension() : f => f));

其中,reducers如下:

    let reducers = combineReducers({AReducer, BReducer,CReducer});//AReducer等是我们自己定义的reducer

那么,当一个type=“A”的action产生后,是如何去这些{AReducer, BReducer,CReducer}来匹配查找然后进行处理的呢?自己当时的猜想是遍历所有的,看了源码之后原来真的是这样。

combineReducers

先看combineReducers的源码里面做了些什么,源码如下:

    export default function combineReducers(reducers) {const reducerKeys = Object.keys(reducers)const finalReducers = {}for (let i = 0; i < reducerKeys.length; i++) {const key = reducerKeys[i]if (process.env.NODE_ENV !== 'production') {if (typeof reducers[key] === 'undefined') {warning(`No reducer provided for key "${key}"`)}}if (typeof reducers[key] === 'function') {finalReducers[key] = reducers[key]}}const finalReducerKeys = Object.keys(finalReducers)let unexpectedKeyCacheif (process.env.NODE_ENV !== 'production') {unexpectedKeyCache = {}}let shapeAssertionErrortry {assertReducerShape(finalReducers)} catch (e) {shapeAssertionError = e}//返回的是如下这个函数,用于处理actionreturn function combination(state = {}, action) {if (shapeAssertionError) {throw shapeAssertionError}if (process.env.NODE_ENV !== 'production') {const warningMessage = getUnexpectedStateShapeWarningMessage(state, finalReducers, action, unexpectedKeyCache)if (warningMessage) {warning(warningMessage)}}let hasChanged = falseconst nextState = {}for (let i = 0; i < finalReducerKeys.length; i++) {const key = finalReducerKeys[i]const reducer = finalReducers[key]const previousStateForKey = state[key]const nextStateForKey = reducer(previousStateForKey, action)if (typeof nextStateForKey === 'undefined') {const errorMessage = getUndefinedStateErrorMessage(key, action)throw new Error(errorMessage)}nextState[key] = nextStateForKeyhasChanged = hasChanged || nextStateForKey !== previousStateForKey}return hasChanged ? nextState : state}}

上面的代码比较长,只需要注意两点即可:

1、将一系列的reducer以(reducerKey,reducer)存储在finalReducers中。

2、返回的函数将在store中自动调用,来处理action。至于如何处理的,下面分析完createStore之后将会分析。

createStore

下面是createStore方法的部分源码,由于store.dispatch(action)是用来分发action,这是修改state的唯一方式,基于此我们这里只关注dispatch方法,因此只对其进行了保留,如果想对其他方法有了解,可以参考博文:Redux从设计到源码,

    export default function createStore(reducer, preloadedState, enhancer) {//省略类型检查      let currentReducer = reducerlet currentState = preloadedStatelet currentListeners = []let nextListeners = currentListenerslet isDispatching = false//省略其他方法,只保留了dispatch方法   function dispatch(action) {//省略了类型检查   try {isDispatching = truecurrentState = currentReducer(currentState, action) //分析} finally {isDispatching = false}const listeners = currentListeners = nextListenersfor (let i = 0; i < listeners.length; i++) {const listener = listeners[i]listener()}return action}}

store.dispatch()方法总结:

1、调用Reducer,传参(currentState,action)。

2、按顺序执行listener。

3、返回action。

下面分析第一点:调用Reducer,传参(currentState,action),即如下这行代码

currentState = currentReducer(currentState, action) 

前面我们说过,currentReducer所指的就是由这行代码let reducers = combineReducers({AReducer, BReducer,CReducer});所产生的reducers,这个reducers是如下这个函数:

    function combination(state = {}, action) {//省略了部分检查代码let hasChanged = falseconst nextState = {}//对所有的reducers进行遍历来处理action。for (let i = 0; i < finalReducerKeys.length; i++) {const key = finalReducerKeys[i]const reducer = finalReducers[key]const previousStateForKey = state[key]//利用这个reducer来处理actionconst nextStateForKey = reducer(previousStateForKey, action)if (typeof nextStateForKey === 'undefined') {const errorMessage = getUndefinedStateErrorMessage(key, action)throw new Error(errorMessage)}//保存state并判断状态是否改变了。nextState[key] = nextStateForKeyhasChanged = hasChanged || nextStateForKey !== previousStateForKey}return hasChanged ? nextState : state} 

上面比较简单哈,看完这里的源码是不是就理解了,当用户操作view产生一个action之后,store是如何自动调用reducers来处理action的哈。

这篇关于Redux记录:Store是如何自动调用reducers来处理action的的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/987666

相关文章

解决docker目录内存不足扩容处理方案

《解决docker目录内存不足扩容处理方案》文章介绍了Docker存储目录迁移方法:因系统盘空间不足,需将Docker数据迁移到更大磁盘(如/home/docker),通过修改daemon.json配... 目录1、查看服务器所有磁盘的使用情况2、查看docker镜像和容器存储目录的空间大小3、停止dock

5 种使用Python自动化处理PDF的实用方法介绍

《5种使用Python自动化处理PDF的实用方法介绍》自动化处理PDF文件已成为减少重复工作、提升工作效率的重要手段,本文将介绍五种实用方法,从内置工具到专业库,帮助你在Python中实现PDF任务... 目录使用内置库(os、subprocess)调用外部工具使用 PyPDF2 进行基本 PDF 操作使用

分析 Java Stream 的 peek使用实践与副作用处理方案

《分析JavaStream的peek使用实践与副作用处理方案》StreamAPI的peek操作是中间操作,用于观察元素但不终止流,其副作用风险包括线程安全、顺序混乱及性能问题,合理使用场景有限... 目录一、peek 操作的本质:有状态的中间操作二、副作用的定义与风险场景1. 并行流下的线程安全问题2. 顺

JAVA实现Token自动续期机制的示例代码

《JAVA实现Token自动续期机制的示例代码》本文主要介绍了JAVA实现Token自动续期机制的示例代码,通过动态调整会话生命周期平衡安全性与用户体验,解决固定有效期Token带来的风险与不便,感兴... 目录1. 固定有效期Token的内在局限性2. 自动续期机制:兼顾安全与体验的解决方案3. 总结PS

MyBatis/MyBatis-Plus同事务循环调用存储过程获取主键重复问题分析及解决

《MyBatis/MyBatis-Plus同事务循环调用存储过程获取主键重复问题分析及解决》MyBatis默认开启一级缓存,同一事务中循环调用查询方法时会重复使用缓存数据,导致获取的序列主键值均为1,... 目录问题原因解决办法如果是存储过程总结问题myBATis有如下代码获取序列作为主键IdMappe

Python异常处理之避免try-except滥用的3个核心原则

《Python异常处理之避免try-except滥用的3个核心原则》在Python开发中,异常处理是保证程序健壮性的关键机制,本文结合真实案例与Python核心机制,提炼出避免异常滥用的三大原则,有需... 目录一、精准打击:只捕获可预见的异常类型1.1 通用异常捕获的陷阱1.2 精准捕获的实践方案1.3

Pandas处理缺失数据的方式汇总

《Pandas处理缺失数据的方式汇总》许多教程中的数据与现实世界中的数据有很大不同,现实世界中的数据很少是干净且同质的,本文我们将讨论处理缺失数据的一些常规注意事项,了解Pandas如何表示缺失数据,... 目录缺失数据约定的权衡Pandas 中的缺失数据None 作为哨兵值NaN:缺失的数值数据Panda

C++中处理文本数据char与string的终极对比指南

《C++中处理文本数据char与string的终极对比指南》在C++编程中char和string是两种用于处理字符数据的类型,但它们在使用方式和功能上有显著的不同,:本文主要介绍C++中处理文本数... 目录1. 基本定义与本质2. 内存管理3. 操作与功能4. 性能特点5. 使用场景6. 相互转换核心区别

Python动态处理文件编码的完整指南

《Python动态处理文件编码的完整指南》在Python文件处理的高级应用中,我们经常会遇到需要动态处理文件编码的场景,本文将深入探讨Python中动态处理文件编码的技术,有需要的小伙伴可以了解下... 目录引言一、理解python的文件编码体系1.1 Python的IO层次结构1.2 编码问题的常见场景二

docker编写java的jar完整步骤记录

《docker编写java的jar完整步骤记录》在平常的开发工作中,我们经常需要部署项目,开发测试完成后,最关键的一步就是部署,:本文主要介绍docker编写java的jar的相关资料,文中通过代... 目录all-docker/生成Docker打包部署文件配置服务A的Dockerfile (a/Docke