Redux学习(二)——封装connect函数

一、自定义connect函数

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

import {PureComponent} from "react";
import store from "../store";
export function connect(mapStateToProps, mapDispatchToProps) {return function enhanceHOC(WrappedComponent) {return class extends PureComponent {constructor(props) {super(props);this.state = {storeState: mapStateToProps(store.getState())}}componentDidMount() {this.unsubscribe = store.subscribe(() => {this.setState({storeState: mapStateToProps(store.getState())})})}componentWillUnmount() {this.unsubscribe()}render() {return <WrappedComponent{...this.props}{...mapStateToProps(store.getState())}{...mapDispatchToProps(store.dispatch)}/>}}}
}

About2.js:

import React from 'react';
import {connect} from "../utils/connect";
import {subAction} from "../store/actionCreator";
function About(props) {return (<div><h1>About</h1><h3>当前计数:{props.counter}</h3><button onClick={e => props.subNumber(1)}>-1</button><button onClick={e => props.subNumber(5)}>-5</button></div>);
}
const mapStateToProps = (state) => {return {counter: state.counter}
}
const mapDispatchToProps = (dispatch) => {return {subNumber: function (num) {dispatch(subAction(num))}}
}
export default connect(mapStateToProps, mapDispatchToProps)(About)

二、自定义context处理store

但是上面的connect函数有一个很大的缺陷:依赖导入的
store
如果我们将其封装成一个独立的库,需要依赖用于创建
的store,我们应该如何去获取呢?
难道让用户来修改我们的源码吗?不太现实;

正确的做法是我们提供一个Provider,Provider来自于我们
创建的Context,让用户将store传入到value中即可;
在这里插入图片描述
在这里插入图片描述

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

import React from "react";const StoreContext = React.createContext()export {StoreContext
}

connect.js:

import {PureComponent} from "react";
// import store from "../store";
import {StoreContext} from './context'
export function connect(mapStateToProps, mapDispatchToProps) {return function enhanceHOC(WrappedComponent) {class EnhanceComponents extends PureComponent {constructor(props, context) {super(props, context);this.state = {storeState: mapStateToProps(context.getState())}}componentDidMount() {this.unsubscribe = this.context.subscribe(() => {this.setState({storeState: mapStateToProps(this.context.getState())})})}componentWillUnmount() {this.unsubscribe()}render() {return <WrappedComponent{...this.props}{...mapStateToProps(this.context.getState())}{...mapDispatchToProps(this.context.dispatch)}/>}}EnhanceComponents.contextType = StoreContextreturn EnhanceComponents}
}

三、react-redux使用

开始之前需要强调一下,redux和react没有直接的关系,你完全可以在React, Angular, Ember, jQuery, or vanilla JavaScript中使用Redux。

尽管这样说,redux依然是和React或者Deku的库结合的更好,因为他们是通过state函数来描述界面的状态,Redux可以发射状态的更新,让他们作出相应。

虽然我们之前已经实现了connect、Provider这些帮助我们完成连接redux、react的辅助工具,但是实际上redux官方帮助我们提供了 react-redux 的库,可以直接在项目中使用,并且实现的逻辑会更加的严谨和高效。

安装react-redux:

yarn add react-redux
或
npm i react-redux --save

在原有代码中做如下的修改:

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

四、组件中异步操作

在之前简单的案例中,redux中保存的counter是一个本地定义的数据

我们可以直接通过同步的操作来dispatch action,state就会被立即更新。

但是真实开发中,redux中保存的很多数据可能来自服务器,我们需要进行异步的请求,再将数据保存到redux中。

在之前学习网络请求的时候我们讲过,网络请求可以在class组件的componentDidMount中发送,所以我们可以有这样的结构:
在这里插入图片描述
我现在完成如下案例操作:
在Home3组件中请求banners和recommends的数据;
在About3组件中展示banners和recommends的数据;
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

Home3.js:

import React, {PureComponent} from 'react';
import {connect} from "react-redux";
import {addAction, changeBannersAction, changeRecommendsAction} from "../store/actionCreator";
import axios from "axios";class Home extends PureComponent {componentDidMount() {axios({url: 'http://123.207.32.32:8000/home/multidata',}).then(res => {console.log(res)const data = res.data.dataconsole.log('轮播图:', data.banner.list)console.log('推荐:', data.recommend.list)this.props.changeBanners(data.banner.list)this.props.changeRecommends(data.recommend.list)})}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));},changeBanners(banners) {dispatch(changeBannersAction(banners));},changeRecommends(recommends) {dispatch(changeRecommendsAction(recommends));}
})export default connect(mapStateToProps, mapDispatchToProps)(Home);

在这里插入图片描述

About3.js:

import React from 'react';
// import {connect} from "../utils/connect";
import {connect} from "react-redux";
import {subAction} from "../store/actionCreator";function About(props) {return (<div><h1>About</h1><h3>当前计数:{props.counter}</h3><button onClick={e => props.subNumber(1)}>-1</button><button onClick={e => props.subNumber(5)}>-5</button><h1>Banner</h1><ul>{props.banners.map((item, index) => {return <li key={item.acm}>{item.title}</li>})}</ul><h1>Recommends</h1><ul>{props.recommends.map((item, index) => {return <li key={item.acm}>{item.title}</li>})}</ul></div>);
}const mapStateToProps = (state) => {return {counter: state.counter,banners: state.banners,recommends: state.recommends}
}
const mapDispatchToProps = (dispatch) => {return {subNumber: function (num) {dispatch(subAction(num))}}
}
export default connect(mapStateToProps, mapDispatchToProps)(About)

五、redux中异步操作

上面的代码有一个缺陷:
我们必须将网络请求的异步代码放到组件的生命周期中来完成;
事实上,网络请求到的数据也属于我们状态管理的一部分,更好的一种方式应该是将其也交给redux来管理;

在这里插入图片描述
但是在redux中如何可以进行异步的操作呢?

答案就是使用中间件(Middleware);
学习过Express或Koa框架的童鞋对中间件的概念一定不陌生;

在这类框架中,Middleware可以帮助我们在请求和响应之间嵌入一些操作的代码,比如cookie解析、日志记录、文件压缩等操作;

六、理解中间件

redux也引入了中间件(Middleware)的概念:

这个中间件的目的是在dispatch的action和最终达到的reducer之间,扩展一些自己的代码;
比如日志记录、调用异步接口、添加代码调试功能等等;

我们现在要做的事情就是发送异步的网络请求,所以我们可以添加对应的中间件:
这里官网推荐的、包括演示的网络请求的中间件是使用 redux-thunk;

redux-thunk是如何做到让我们可以发送异步的请求呢?

  1. 我们知道,默认情况下的dispatch(action),action需要是一个JavaScript的对象;

  2. redux-thunk可以让dispatch(action函数),action可以是一个函数

  3. 该函数会被调用,并且会传给这个函数一个dispatch函数和getState函数;

  • dispatch函数用于我们之后再次派发action;
  • getState函数考虑到我们之后的一些操作需要依赖原来的状态,用于让我们可以获取之前的一些状态;

七、如何使用redux-thunk

  1. 安装redux-thunk
yarn add redux-thunk
  1. 在创建store时传入应用了middleware的enhance函数
    通过applyMiddleware来结合多个Middleware, 返回一个enhancer;
    将enhancer作为第二个参数传入到createStore中;
    在这里插入图片描述
    在这里插入图片描述

  2. 定义返回一个函数的action:
    注意:这里不是返回一个对象了,而是一个函数;
    该函数在dispatch之后会被执行;
    在这里插入图片描述

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

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

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

相关文章

python ssh实现_SSH协议的Python实现paramiko

paramiko安装SSH是一个协议&#xff0c;paramiko 是一个Python 的库&#xff0c;该库支持sshv2协议&#xff0c;实现了对远程服务器执行操作安装命令&#xff1a;pip3 install paramikoSSHClient类与SFTPClient类paramiko包含两个核心组建&#xff0c;分别是SSHClient和SFTPCli…

C++学习10 static静态成员变量和静态成员函数

一般情况下&#xff0c;如果有N个同类的对象&#xff0c;那么每一个对象都分别有自己的成员变量&#xff0c;不同对象的成员变量各自有值&#xff0c;互不相干。但是有时我们希望有某一个或几个成员变量为所有对象共有&#xff0c;这样可以实现数据共享。 可以使用全局变量来达…

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

一、redux-devtools 我们之前讲过&#xff0c;redux可以方便的让我们对状态进行跟踪和调试&#xff0c;那么如何做到呢&#xff1f; redux官网为我们提供了redux-devtools的工具&#xff1b;利用这个工具&#xff0c;我们可以知道每次状态是如何被修改的&#xff0c;修改前后…

python 存储图片 alpha_保存时Matplotlib图形面颜色alpha(背景色、透明度)

前一个问题是使用savefig()保存屏幕上显示的相同的面颜色(背景色)&#xff0c;即&#xff1a;fig plt.figure()fig.patch.set_facecolor(blue)fig.savefig(foo.png, facecolorfig.get_facecolor())(使用savefig()需要重新指定背景色。)fig.patch.set_alpha(0.5)我找不到一种方…

【JavaScript代码实现四】获取和设置 cookie

1 // 创建cookie2 function setCookie(name, value, expires, path, domain, secure) {3 var cookieText encodeURIComponent(name) encodeURIComponent(value);4 if (expires instanceof Date) {5 cookieText ; expires expires; 6 }7 if…

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

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

Linux中source是什么指令?

命令用法&#xff1a; source FileName作用&#xff1a;在当前bash环境下读取并执行FileName中的命令。 注&#xff1a;该命令通常用命令“.”来替代。 如&#xff1a;source /etc/profile 与 . /etc/profile 是等效的。 注意&#xff1a;source命令与shell scripts的区别是&a…

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…

Redis 3.0.2集群搭建以及相关问题汇总

Redis3 正式支持了 cluster&#xff0c;是为了解决构建redis集群时的诸多不便&#xff08;1&#xff09;像操作单个redis一样操作key&#xff0c;不用操心key在哪个节点上&#xff08;2&#xff09;在线动态添加、删除redis节点&#xff0c;不用停止服务&#xff08;3&#xff…

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…

python长沙_长沙python

本文转自量子位(ID:QbitAI) 边策 鱼羊 发自 凹非寺 量子位 报道 | 公众号 QbitAI 只用99行代码&#xff0c;你也可以像《冰雪奇缘》里的艾莎公主一样拥有冰雪魔法。虽然你不能在现实世界中肆意变出魔法&#xff0c;但却能在计算机的虚拟世界挥洒特效。或许你不知道&#xff0c;…

sed

sed -n 10p text.txt 输出第10行转载于:https://www.cnblogs.com/ssyuxue/p/5804857.html

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博客​…

springContext

方法一 package com.hsh.utils; import org.springframework.beans.BeansException;import org.springframework.context.ApplicationContext;import org.springframework.context.ApplicationContextAware;import org.springframework.stereotype.Component;Componentpublic c…

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

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

python制作聊天机器人原理_用 Python 来做一个聊天机器人吧!(一)

在我的一个回答里&#xff0c;我提到了用 Python 搭建聊天机器人。从今天开始&#xff0c;我就带着大家从0开始搭建一个聊天机器人。(顺便说一句&#xff0c;我喜欢把链接像上面这样加在文字里&#xff0c;如果找不到文中所说的资源&#xff0c;可以看看周围有没有链接。)准备工…