Redux学习(三)——redux-saga的使用、编写中间件函数、Reducer文件拆分

一、redux-devtools

我们之前讲过,redux可以方便的让我们对状态进行跟踪和调试,那么如何做到呢?

  • redux官网为我们提供了redux-devtools的工具;
  • 利用这个工具,我们可以知道每次状态是如何被修改的,修改前后的状态变化等等;

在这里插入图片描述

安装该工具需要两步:

  1. 第一步:在对应的浏览器中安装相关的插件(比如Chrome浏览器扩展商店中搜索Redux DevTools即可,其他方法可以参考GitHub);
  2. 第二步:在redux中继承devtools的中间件;

在这里插入图片描述
index.js:

import {createStore, applyMiddleware, compose} from 'redux'
import reducer from "./reducer.js";
import thunkMiddleware from 'redux-thunk'// composeEnhancers函数
const composeEnhancers = window.__REDUX_DEVTOOLS_EXTENSION_COMPOSE__({trace: true}) || compose;// 通过applyMiddleware来结合多个Middleware,返回一个storeEnhancer
const storeEnhancer = applyMiddleware(thunkMiddleware)
const store = createStore(reducer, composeEnhancers(storeEnhancer))export default store;

二、generator

saga中间件使用了ES6的generator语法,所以我们有必须简单讲解一下:
注意:我这里并没有列出generator的所有用法,事实上它的用法非常的灵活,大家可以自行去学习一下。

我们按照如下步骤演示一下生成器的使用过程:
在JavaScript中编写一个普通的函数,进行调用会立即拿到这个函数的返回结果。

如果我们将这个函数编写成一个生成器函数。调用iterator的next函数,会销毁一次迭代器,并且返回一个yield的结果。
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

研究一下foo生成器函数代码的执行顺序
generator和promise一起使用:
在这里插入图片描述

   // generator 和 Promise一起使用function* bar() {console.log(111)const result = yield new Promise((resolve, reject) => {setTimeout(() => {resolve('Hello Generator')}, 3000)})console.log(result)}const it = bar()// console.log(it.next().value); // Promise对象it.next().value.then((res) => {it.next(res)})

三、redux-saga的使用

redux-saga是另一个比较常用在redux发送异步请求的中间件,它的使用更加的灵活。

Redux-saga的使用步骤如下

  1. 安装redux-saga
    yarn add redux-saga
  2. 集成redux-saga中间件
    导入创建中间件的函数;
    通过创建中间件的函数,创建中间件,并且放到applyMiddleware函数中;
    启动中间件的监听过程,并且传入要监听的saga;

store/index.js:

import {createStore, applyMiddleware, compose} from 'redux'
import reducer from "./reducer.js";
import thunkMiddleware from 'redux-thunk'
import createSagaMiddleware from 'redux-saga'
import saga from './saga'// composeEnhancers函数
const composeEnhancers = window.__REDUX_DEVTOOLS_EXTENSION_COMPOSE__({trace: true}) || compose;// 通过applyMiddleware来结合多个Middleware,返回一个storeEnhancer
// 1.引入thunkMiddleware中间件(代码在上面)
// 2.创建sagaMiddleware中间件
const sagaMiddleware = createSagaMiddleware()const storeEnhancer = applyMiddleware(thunkMiddleware, sagaMiddleware)
const store = createStore(reducer, composeEnhancers(storeEnhancer))sagaMiddleware.run(saga) // saga为生成器函数export default store;

在这里插入图片描述
3. saga.js文件的编写

takeEvery:可以传入多个监听的actionType,每一个都可以被执行(对应有一个takeLatest,会取消前面的)

put:在saga中派发action不再是通过dispatch,而是通过put;

all:可以在yield的时候put多个action;

在这里插入图片描述
store/saga.js:

import {takeEvery, put, all, takeLatest} from 'redux-saga/effects'
import axios from "axios";
import {FETCH_HOME_MULTIDATA} from "./constants";
import {changeBannersAction,changeRecommendsAction} from "./actionCreator";
function* fetchHomeMultidata(action) {const res = yield axios.get('http://123.207.32.32:8000/home/multidata')const banners = res.data.data.banner.listconst recommends = res.data.data.recommend.list// yield put(changeBannersAction(banners))// yield put(changeRecommendsAction(recommends))yield all([yield put(changeBannersAction(banners)),yield put(changeRecommendsAction(recommends))])
}
function* mySaga() {// takeEvery: 每个action都会被执行// yield takeEvery(FETCH_HOME_MULTIDATA, fetchHomeMultidata)// takeLatest:一次只能监听一个对应的action// yield takeLatest(FETCH_HOME_MULTIDATA, fetchHomeMultidata)yield all([yield takeEvery(FETCH_HOME_MULTIDATA, fetchHomeMultidata)// yield takeLatest(FETCH_HOME_MULTIDATA, fetchHomeMultidata)])
}
export default mySaga

在这里插入图片描述
home.js:

import React, {PureComponent} from 'react';
import {connect} from "react-redux";
import {addAction,fetchHomeMultidataAction
}from "../store/actionCreator";class Home extends PureComponent {componentDidMount() {this.props.getHomeMultidata()}render() {return (<div><h1>Home</h1><h3>当前计数:{this.props.counter}</h3><button onClick={e => this.props.increment()}>+1</button><button onClick={e => this.props.addNumber(5)}>+5</button></div>);}
}
const mapStateToProps = state => ({counter: state.counter
})
const mapDispatchToProps = dispatch => ({increment() {dispatch(addAction(1));},addNumber(num) {dispatch(addAction(num));},getHomeMultidata() {dispatch(fetchHomeMultidataAction)}
})
export default connect(mapStateToProps, mapDispatchToProps)(Home);

在这里插入图片描述
在这里插入图片描述

四、打印日志需求

前面我们已经提过,中间件的目的是在redux中插入一些自己的操作:

  • 比如我们现在有一个需求,在dispatch之前,打印一下本次的action对象,dispatch完成之后可以打印一下最新的store
    state;
  • 也就是我们需要将对应的代码插入到redux的某部分,让之后所有的dispatch都可以包含这样的操作;

如果没有中间件,我们是否可以实现类似的代码呢? 可以在派发的前后进行相关的打印。
但是这种方式缺陷非常明显:

  • 首先,每一次的dispatch操作,我们都需要在前面加上这样的逻辑代码;
  • 其次,存在大量重复的代码,会非常麻烦和臃肿;

是否有一种更优雅的方式来处理这样的相同逻辑呢?

  • 我们可以将代码封装到一个独立的函数中

但是这样的代码有一个非常大的缺陷:

  • 调用者(使用者)在使用我的dispatch时,必须使用我另外封装的一个函数dispatchAndLog;
  • 显然,对于调用者来说,很难记住这样的API,更加习惯的方式是直接调用dispatch;

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

修改dispatch

事实上,我们可以利用一个hack一点的技术:Monkey Patching,利用它可以修改原有的程序逻辑;
我们对代码进行如下的修改:

  • 这样就意味着我们已经直接修改了dispatch的调用过程;
  • 在调用dispatch的过程中,真正调用的函数其实是dispatchAndLog;

当然,我们可以将它封装到一个模块中,只要调用这个模块中的函数,就可以对store进行这样的处理:
在这里插入图片描述

thunk需求

redux-thunk的作用:
我们知道redux中利用一个中间件redux-thunk可以让我们的dispatch不再只是处理对象,并且可以处理函数;那么redux-thunk中的基本实现过程是怎么样的呢?事实上非常的简单。

我们来看下面的代码:
我们又对dispatch进行转换,这个dispatch会判断传入的
在这里插入图片描述

合并中间件

单个调用某个函数来合并中间件并不是特别的方便,我们可以封装一个函数来实现所有的中间件合并:
在这里插入图片描述
我们来理解一下上面操作之后,代码的流程:
在这里插入图片描述
当然,真实的中间件实现起来会更加的灵活,这里我们仅仅做一个抛砖引玉,有兴趣可以参考redux合并中间件的源码流程。

五、Reducer代码拆分

我们先来理解一下,为什么这个函数叫reducer?
我们来看一下目前我们的reducer:

  • 当前这个reducer既有处理counter的代码,又有处理home页面的数据;
  • 后续counter相关的状态或home相关的状态会进一步变得更加复杂;
  • 我们也会继续添加其他的相关状态,比如购物车、分类、歌单等等;
  • 如果将所有的状态都放到一个reducer中进行管理,随着项目的日趋庞大,必然会造成代码臃肿、难以维护。

因此,我们可以对reducer进行拆分:

  • 我们先抽取一个对counter处理的reducer;
  • 再抽取一个对home处理的reducer;
  • 将它们合并起来;

六、Reducer文件拆分

目前我们已经将不同的状态处理拆分到不同的reducer中,我们来思考:

  • 虽然已经放到不同的函数了,但是这些函数的处理依然是在同一个文件中,代码非常的混乱;
  • 另外关于reducer中用到的constant、action等我们也依然是在同一个文件中;
    在这里插入图片描述
    在这里插入图片描述
    在这里插入图片描述

在这里插入图片描述

七、combineReducers函数

目前我们合并的方式是通过每次调用reducer函数自己来返回一个新的对象。
事实上,redux给我们提供了一个combineReducers函数可以方便的让我们对多个reducer进行合并:
在这里插入图片描述
那么combineReducers是如何实现的呢?
事实上,它也是讲我们传入的reducers合并到一个对象中,最终返回一个combination的函数(相当于我们之前的reducer函数了);

在执行combination函数的过程中,它会通过判断前后返回的数据是否相同来决定返回之前的state还是新的state; 新的state会触发订阅者发生对应的刷新,而旧的state可以有效的组织订阅者发生刷新;

八、React中的state如何管理

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

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

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

相关文章

react-router的使用(一)——URL的hash、HTML5的history、Router的基本使用

一、阶段一&#xff1a;后端路由阶段 早期的网站开发整个HTML页面是由服务器来渲染的. 服务器直接生产渲染好对应的HTML页面, 返回给客户端进行展示. 但是, 一个网站, 这么多页面服务器如何处理呢? 一个页面有自己对应的网址, 也就是URL.URL会发送到服务器, 服务器会通过正…

react-router的使用(二)——NavLink的使用、Switch的作用、Redirect

一、NavLink的使用 需求&#xff1a;路径选中时&#xff0c;对应的a元素变为红色 这个时候&#xff0c;我们要使用NavLink组件来替代Link组件&#xff1a; activeStyle&#xff1a;活跃时&#xff08;匹配时&#xff09;的样式&#xff1b;activeClassName&#xff1a;活跃时…

群晖ffmpeg_群晖Video station支持DTS和EAC3

群晖video station这个套件现在经过群晖的打磨&#xff0c;现在还是不错的&#xff0c;支持硬件解码和蓝光等多媒体播放&#xff0c;比起PLEX和EMBY动辄好几百的会员费&#xff0c;这个免费的用起来还真香&#xff0c;但是因为种种小问题需要解决了&#xff0c;才能好用&#x…

react-router的使用(三)——路由的嵌套

一、路由的嵌套 在开发中&#xff0c;路由之间是存在嵌套关系的。 这里我们假设about页面中有三个页面内容&#xff1a; 企业历史、企业文化和联系我们&#xff1b;点击不同的链接可以跳转到不同的地方&#xff0c;显示不同的内容&#xff1b; 二、手动路由跳转 目前我们实现…

如何阅读一本书 pdf_《如何阅读一本书》:一本书,四个层次,看阅读小白如何逆袭?...

“读书不是为了雄辩和驳斥&#xff0c;也不是为了轻信和盲从&#xff0c;而是为了思考和权衡。”这是培根的一句名言&#xff0c;我们都曾经被这样的读书警句激励的斗志昂扬&#xff0c;于是立下目标一年或是一个月要读多少本书&#xff0c;结果发现一切是徒劳。你是否曾经斗志…

React Hooks的使用(一)——useState、useEffect解析

一、为什么需要Hook? Hook 是 React 16.8 的新增特性&#xff0c;它可以让我们在不编写class的情况下使用state以及其他的React特性&#xff08;比如生命周期&#xff09;。 我们先来思考一下class组件相对于函数式组件有什么优势&#xff1f;比较常见的是下面的优势&#xf…

React Hooks的使用(二)——useContext、useReducer、useCallback、useMemo解析

一、useContext的使用 在之前的开发中&#xff0c;我们要在组件中使用共享的Context有两种方式&#xff1a; 第一种方式&#xff1a;类组件可以通过 类名.contextType MyContext方式&#xff0c;在类中获取context&#xff1b; 第二种方式&#xff1a;多个Context或者在函数…

凸多边形面积_C++计算任意多边形的面积

任意多边形的面积计算_拾忆楓灵的博客-CSDN博客​blog.csdn.net计算任意多边形的面积 - tenos - 博客园​www.cnblogs.com完美解决计算3D空间任意多边形面积_Studiouss的博客-CSDN博客​blog.csdn.net求多边形面积公式&#xff08;已知顶点坐标&#xff09;_扬帆起航-CSDN博客​…

React Hooks的使用(三)——useRef、useImperativeHandle、useLayoutEffect解析、自定义Hook

一、useRef useRef返回一个ref对象&#xff0c;返回的ref对象再组件的整个生命周期保持不变。 最常用的ref是两种用法&#xff1a; 用法一&#xff1a;引入DOM&#xff08;或者组件&#xff0c;但是需要是class组件&#xff09;元素&#xff1b; 案例一&#xff1a;引用DOM …

python子类分配

原问题是将左边样式变成右边样式&#xff1a; 即有父类和子类&#xff0c;父类包括多个子类&#xff0c;怎样将子类匹配到父类下面的问题 代码如下 1 #!/usr/bin/python3.42 # -*- coding: utf-8 -*-3 4 arr1 ["S01","S01","S01","S02&quo…

react 网易云音乐实战项目笔记

0、项目规范 一、路由相关 npm i react-router-dom npm i react-router-config // 用于配置路由映射的关系数组1. 路由重定向 访问 /路径 》 重定向到 /discover路径 2. 二级路由&#xff1a; 二、网络请求相关 npm i axios页面中使用暴露 出来的request发送网络请求&#…

react项目打包

一、react项目打包 对于使用脚手架创建的项目&#xff0c;打包是一件非常容易的事情&#xff1a; yarn build其他文件没有太多要解析的&#xff0c;我们看一下js文件&#xff1a; [hash].chunk.js 代表是所有依赖的第三方库&#xff0c; vendor(第三方库) 的代码&#xff1b…

React SSR

一、为什么需要SSR呢&#xff1f; 单页面富应用的局限&#xff1a; 之前我们开发的应用程序&#xff0c;如果直接请求可以看到上面几乎没有什么内容。 但是为什么我们页面可以看到大量的内容呢&#xff1f; 因为当我们请求下来静态资源之后会执行JS&#xff0c;JS会去请求数…

Vue3 组件通信学习笔记

一、父子组件之间通信 父子组件之间如何进行通信呢&#xff1f; 父组件传递给子组件&#xff1a;通过props属性&#xff1b;子组件传递给父组件&#xff1a;通过$emit触发事件&#xff1b; 1.1 父组件传递给子组件 在开发中很常见的就是父子组件之间通信&#xff0c;比如父…

Vue3 slot插槽——(默认插槽、具名插槽、作用域插槽)

一、认识插槽Slot 在开发中&#xff0c;我们会经常封装一个个可复用的组件&#xff1a; 前面我们会通过props传递给组件一些数据&#xff0c;让组件来进行展示&#xff1b;但是为了让这个组件具备更强的通用性&#xff0c;我们不能将组件中的内容限制为固定的div、span等等这…

Vue动态组件和组件缓存

一、切换组件案例 比如我们现在想要实现了一个功能&#xff1a; 点击一个tab-bar&#xff0c;切换不同的组件显示&#xff1b; 这个案例我们可以通过两种不同的实现思路来实现&#xff1a; 方式一&#xff1a;通过v-if来判断&#xff0c;显示不同的组件&#xff1b; 方式二…

Webpack的代码分包Vue3中定义异步组件分包refs的使用

一、默认的打包过程&#xff1a; 默认情况下&#xff0c;在构建整个组件树的过程中&#xff0c;因为组件和组件之间是通过模块化直接依赖的&#xff0c;那么webpack在打包时就会将组件模块打包到一起&#xff08;比如一个app.js文件中&#xff09;&#xff1b; 这个时候随着项…

组件的v-model Mixin extends

一、组件的v-model 前面我们在input中可以使用v-model来完成双向绑定&#xff1a; 这个时候往往会非常方便&#xff0c;因为v-model默认帮助我们完成了两件事&#xff1b;v-bind:value的数据绑定 和 input的事件监听&#xff1b; 如果我们现在封装了一个组件&#xff0c;其…

Vue3过渡动画实现

一、认识动画 在开发中&#xff0c;我们想要给一个组件的显示和消失添加某种过渡动画&#xff0c;可以很好的增加用户体验&#xff1a; React框架本身并没有提供任何动画相关的API&#xff0c;所以在React中使用过渡动画我们需要使用一个第三方库react-transition-group&…

Vue3 Composition API(一)——setup、reactive、ref、readonly

一、Options API的弊端 在Vue2中&#xff0c;我们编写组件的方式是Options API&#xff1a; Options API的一大特点就是在对应的属性中编写对应的功能模块&#xff1b;比如data定义数据、methods中定义方法、computed中定义计算属性、watch中监听属性改变&#xff0c;也包括生…