React整理总结(五、Redux)

news/2024/7/15 18:28:17 标签: react.js, javascript, 前端

1.Redux核心概念

纯函数
  • 确定的输入,一定会产生确定的输出;
  • 函数在执行过程中,不能产生副作用
store

存储数据

action

更改数据

reducer

连接store和action的纯函数
将传入的state和action结合,生成一个新的state

  • dispatch派发action修改store
  • subscribe | unsubscribe订阅store的数据发生变化
javascript">// store/index.js
import { createStore } from 'redux';

const initState = {
	msg: "hello redux"
}
/*** 定义reducer,纯函数*****/
function reducer(state = initState, action){
	if (action.type === "change"){
		return {...state, msg: action.payload.msg};
	}
	return state;
}

export default const store = createStore(reducer);


// store/ actionCreator.js
/**** 动态生成action *****/
export const CHANGEMSGACTION = msg => ({type: 'change', payload: {msg}});

// 使用的地方
import store from "~/store";
const unsubscribe = store.subscribe(() => {
	console.log("::::STORE", store.getState());
})
unsubscribe();

// 修改store中的数据
const MSGAction = {
	type: "change", 
	payload: {
		msg: "hello change",
	}
};
store.dispatch(MSGAction);
  • combineReducer将多个reducer合并为一个reducer,达到拆分store的目的

2. Redux三大原则

单一数据源
  • 整个应用程序的state被存储在一颗object tree中,并且这个object tree只存储在一个 store中:
  • R-edux并没有强制让我们不能创建多个Store,但是那样做并不利于数据的维护;
  • 单一的数据源可以让整个应用程序的state变得方便维护、追踪、修改;
State是只读的
  • 唯一修改State的方法一定是触发action,不要试图在其他地方通过任何的方式来修改State:
  • 这样就确保了View或网络请求都不能直接修改state,它们只能通过action来描述自己想要如何修改state;
  • 这样可以保证所有的修改都被集中化处理,并且按照严格的顺序来执行,所以不需要担心race condition(竟态)的问题;
使用纯函数来执行修改
  • 通过reducer将 旧state和 actions联系在一起,并且返回一个新的State:
  • 随着应用程序的复杂度增加,我们可以将reducer拆分成多个小的reducers,分别操作不同state tree的一部分;
  • 但是所有的reducer都应该是纯函数,不能产生任何的副作用;
    在这里插入图片描述

3.react-redux的使用

  • 通过provider给整个app提供store
javascript">// App.jsx
import { Provider } from 'react-redux';
import store from '~/store';

const root = document.querySelector("#root");
root.render(
	<Provider store={store}>
		<App/>
	</Provider>
)
  • 通过connect将组件和store连接。connect会返回一个高阶组件,接受的参数将store中的部分数据映射到组件
javascript">// 组件中
import React, { PureComponent } from "react";
import { connect } from "react-redux";

class MyComp extends PureComponent{
	render(){
		const { msg, changeMsg } = this.props;
		return (<div>
		<h2>{msg}</h2>
		<input onChange={val => changeMsg(val)} />
		</div>
	}
}
/**** 将state映射到props,组件中props中就会有msg ****/
function mapStateToProps(state){
	return {
		msg: state.msg
	}
}
/*** 将修改store的函数添加到组件的props中 ***/
function mspDispatchToProps(dispatch){
	return {
		changeMsg(msg){
			dispatch(CHANGEMSGACTION(msg));
		}
	}
}

export default connect(mapStateToProps, mapDispatchToProps)(MyComp);

  • 异步action—中间件
    • Middleware可以帮助我们在请求和响应之间嵌入一些操作的代码,比如cookie解析、日志记录、文件压缩等操作
    • createStore的第二个参数接受一个中间件,使用react-thunk使得dispatch可以派发函数,在派发的函数中可以异步更新store。
javascript">import { createStore, applyMiddleware } from 'redux';
import thunk from 'redux-thunk';

const reducer = (state, action) => {
	...
	return state;
}

const store = createStore(reducer, applyMiddleware(thunk));

// actionCreator.js
...
/**** 被派发的函数,需要返回一个函数,该函数接受两个参数dispatch,getState,*****/
const fetchHomeDataAction = () => {

	return (dispatch, getState) => {
		fetch(url).then(res => {
			dispatch(HOMEDATAACTION(res.data));
		});
	}
}
// 组件中
function mapDispatchToProps(dispatch){
	return {
		fetchHomeData(){
			dispatch(fetchHomeDataAction()); // 执行action函数
		}
	}
}
  • 使用redux-thunk
    • 在创建store时传入应用了middleware的enhance函数
      通过applyMiddleware来结合多个Middleware, 返回一个enhancer;将enhancer作为第二个参数传入到createStore中;
    • 定义返回一个函数的action:
      这里不是返回一个对象了,而是一个函数;该函数在dispatch之后会被执行;

4.Redux/toolkit

npm install @reduxjs/toolkit react-redux

  • createSlice({name, initialState, reducers:{}})接受reducer函数的对象、切片名称和初始状态值,并自动生成切片reducer,并带有相应的actions。
    • name:用户标记slice的名词, 在之后的redux-devtool中会显示对应的名词;
    • initialState:初始化值. 第一次初始化时的值;
    • reducers:相当于之前的reducer函数.对象类型,并且可以添加很多的函数;函数类似于redux原来reducer中的一个case语句;
      • 参数一:state
      • 参数二:调用这个action时,传递的action参数;
javascript">import { createSlice } from '@reduxjs/toolkit';
const CounterSlice = createSlice({
	name: "counter",
	initialState: {
		count: 0,
	},
	reducers: {
		addNumber(state, action){
			state.counter += action.payload;
		}
	}
});

export const { addNumber } = CounterSlice.action;
export default CounterSlice.reducer;
  • configureStore包装createStore以提供简化的配置选项和良好的默认值。它可以自动组合你的 slice reducer,添加你提供的任何 Redux 中间件,redux-thunk默认包含,并启用 Redux DevTools Extension。
javascript">const store = configureStore({
	reducer: {
		counter: counterReducer;
	}
})
  • createAsyncThunk接受一个动作类型字符串和一个返回承诺的函数,并生成一个pending/fulfilled/rejected基于该承诺分派动作类型的 thunk
javascript">const AXIOSDataSlice = createSlice({
	name: 'axiosdata',
	initialState: {
		data: []
	},
	reducers: {
		setData(state, action){
			state.data = action.payload;
		}
	},
	extraReducers: {
		/**
		[AxiosMultidataAction.pending](state, action){
			state.data = action.payload;
		}
		[AxiosMultidataAction.rejected](state, action){
			state.data = action.payload;
		}
		**/
		[AxiosMultidataAction.fulfilled](state, action){
			state.data = action.payload;
		}
	}
	/*** 链式写法 *****/
	extraReducers: (builder) => {
		builder.addCase(AxiosMultidataAction.pending, (state, action) => {
			console.log("pending");	
		}).addCase(AxiosMultidataAction.fulfilled, (state, action) => {})
	}
})
export default AXIOSDataSlice.reducer;


const AxiosMultidataAction = createAsyncThunk("axiosdata", async (extraInfo, store) => {
// 第一个can
	const res = await getData();
	return res;
})
  • immerjs库保持数据不可变(持久化数据

5 手写connect

javascript">function connect(mapStateToProps, mapDispatchToProps){
	function hoc(Component){
		class HOCComponent extends PureComponent{
			constuctor(props){
				super(props);
				this.state = mapStateToProps(store.getState());
			}
			componentDidMount(){
				this.unsubscribe = store.subscribe(() => {
					//this.forceUpdate();
					this.setState(mapStateToProps(store.getState());
				})
			}
			componentWillUnmount() {
				this.unsubscribe();
			}
			
			render(){
				return <Component {...this.props} {...mapStateToProps(store.getState())} {...mapDispatchToProps(store.dispatch)} />
			}
		}
		return HOCComponent;
	}
	return hoc;
}

http://www.niftyadmin.cn/n/5198706.html

相关文章

STM32:OLED屏幕开发

一、OLED原理 所谓的屏幕就是由一个个小灯组成&#xff0c;每个小灯称之为一个像素。只要在屏幕上有选择地点亮一部分小灯&#xff0c;就可以显示我们想要的图案。所谓下分辨率就是屏幕上的小灯数量。常见单片机中常见的屏幕分辨率常见的就是128(列长)*64(行高)。如果每个小灯都…

C#8.0本质论第十四章--事件

C#8.0本质论第十四章–事件 委托本身是一个更大的模式(Pattern)的基本单位&#xff0c;称为Publish-Subscribe(发布-订阅)或Observer(观察者)。 14.1使用多播委托实现Publish-Subscribe模式 14.1.1定义订阅者方法 public class Cooler {public Cooler(float temperature){T…

Spark SQL将Hive表中的数据写入到MySQL数据库中

import org.apache.spark.sql.SparkSessionobject HiveToMySQL {def main(args: Array[String]): Unit {// 创建SparkSessionval spark SparkSession.builder().appName("HiveToMySQL").enableHiveSupport().getOrCreate()// 读取Hive表数据val hiveDF spark.tabl…

docker swarm集群部署

文章目录 前言一、安装docker1.1 解压1.2 配置docker 存储目录和dns1.3 添加docker.service文件1.4 docker 启动验证 二、docker swarm 集群配置2.1 关闭selinux2.2 设置主机名称并加入/etc/hosts2.3 修改各个服务器名称&#xff08;uname -a 进行验证&#xff09;2.4 初始化sw…

MR混合现实教学系统在汽车检修与维护课堂教学中的应用

传统的汽车检修与维护课堂教学主要依赖教师口头讲解和黑板演示&#xff0c;这种方式存在一定的局限性。首先&#xff0c;对于一些复杂的机械结构和操作过程&#xff0c;教师难以生动形象地展示给学生。其次&#xff0c;学生无法直接观察到实际操作中的细节和注意事项&#xff0…

【Java程序员面试专栏 算法训练篇】二叉树高频面试算法题

一轮的算法训练完成后,对相关的题目有了一个初步理解了,接下来进行专题训练,以下这些题目就是二叉树相关汇总的高频题目 遍历二叉树 遍历二叉树,分为递归和迭代两种方式,递归类似于DFS,迭代类似于BFS,【算法训练-二叉树 一】【遍历二叉树】前序遍历、中序遍历、后续遍…

Flask笔记一之项目搭建、配置项导入

本文首发于公众号&#xff1a;Hunter后端 原文链接&#xff1a;Flask笔记一之项目搭建、配置项导入 这一篇开始介绍 Flask 系列笔记&#xff0c;这个系列笔记将和之前的 Django 笔记一样会从 Flask 的官方文档中提取一系列的知识点&#xff0c;整理成系列笔记。 这是 Flask 系…

【技巧】Word和Excel如何互相转换?

Word文档里有数据表格&#xff0c;如果编辑修改起来感觉没那么方便或容易出错&#xff0c;不妨将文档转换成Excel表格再来处理。 将Word文档转换成Excel&#xff0c;比较常用的是复制粘贴方法&#xff0c;也就是将Word文档的表格复制后&#xff0c;再粘贴到Excel表格里&#x…