React项目的状态管理:Redux Toolkit

目录

1、搭建环境

2、Redux Toolkit 包含了什么

 3、使用示例

(1)创建user切片

(2)合并切片得到store

(3)配置store和使用store


使用js来编写代码,方便理解一些

1、搭建环境

首先,先要新建一个react项目,有了react项目之后就安装好依赖

npm install @reduxjs/toolkit

npm install react-redux

 然后在src文件目录下新建一个名为store/stores文件夹(理论上其他也行,所有开发者都约定好即可),有关状态管理的数据都会存放在这里

这只是一个实例,store这个命名算是约定熟成,虽然可以起个别的名字,但是为了方便团队间合作约定写成store更容易让代码更好维护管理以及新开发者迅速找到对应位置。跟components文件夹放组件一样的道理。

2、Redux Toolkit 包含了什么

在中文版官方文档中可以直接看到以下的话语

看起来有点难懂,但是没关系我们先知道一些知识就好了,接下来是redux toolkit使用示例 

 3、使用示例

(1)创建user切片

在前面写好的store文件下新建一个user.js文件夹:

主要使用的是createSlice

import { createSlice } from "@reduxjs/toolkit"const useSlice = createSlice({name: "user",initialState: { // 初始状态name: "张三",age: 18,},reducers: { // 定义方法setName (state, action) {state.name = action.payload},setAge(state, action) {state.age = action.payload}}
})export const { setName, setAge } = useSlice.actions // 导出方法
export default useSlice.reducer // 导出reducer

在react的状态当中,有一个规则是数据不可变,大白话讲就是这个state不应该直接修改,所以在代码中会直接生成一个新数据去覆盖它。遵循这个规则的话就是会写成setAge这个样子:

import { createSlice } from "@reduxjs/toolkit"const useSlice = createSlice({name: "user",initialState: { // 初始状态name: "张三",age: 18,},reducers: { // 定义方法setName (state, action) {state.name = action.payload},setAge(state, action) {return {...state,age: action.payload}}}
})export const { setName, setAge } = useSlice.actions // 导出方法
export default useSlice.reducer // 导出reducer

redux toolkit支持上面两种写法,所以写哪一种看个人爱好了

还可以继续写不同的文件在store里,对状态进行分类存放,对不同的分类都称为切片

在这里写一下,为什么在user.js要有以下两行代码:

export const { setName, setAge } = useSlice.actions // 导出方法
export default useSlice.reducer // 导出reducer

 直观感受就是很多余,为什么需要这样编写

1. export const { setName, setAge } = useSlice.actions

  • 作用:导出该 slice 的 action creators(setName 和 setAge),供组件或其他地方调用。

  • 为什么需要单独导出?

    • 在组件中,你需要通过 dispatch(setName("Alice")) 这样的方式来触发状态更新。如果这里不导出这些 actions,外部代码就无法使用它们。

    • 虽然可以直接通过 useSlice.actions.setName 访问,但直接解构并导出会更清晰,方便调用。

        示例:

        // 在其他组件中
        import { setName } from './userSlice';
        dispatch(setName("Bob")); // 直接使用导出的 action

2. export default useSlice.reducer

  • 作用:导出 slice 的 reducer,供 Redux store 合并使用。

  • 为什么需要默认导出?

    • Redux store 需要组合所有 slice 的 reducer(比如通过 combineReducers)。通常会在 store 的配置文件中导入各个 slice 的 reducer:

      // store.js
      import userReducer from './userSlice'; // 这里导入的就是 useSlice.reducer
      const rootReducer = combineReducers({user: userReducer,// 其他 reducer...
      });
    • 默认导出是为了简化导入语法(import userReducer from './user'),符合 Redux 的约定俗成

如果不这样写会怎样?

  • 如果不导出 actions,外部代码需要直接访问 useSlice.actions.setName,这会增加耦合性(调用方需要知道 slice 的内部结构)。

  • 如果不默认导出 reducer,store 配置时需要写更长的路径(比如 import { reducer as userReducer } from './user'),不够简洁

总结

这两行代码是 Redux Toolkit 的标准模式:

  1. 导出 actions:方便组件调用

  2. 导出 reducer:方便 store 组合

虽然看起来冗余,但这是为了职责分离和代码可维护性。如果项目中有多个 slice,这种模式能保持一致性,减少认知负担

(2)合并切片得到store

将前面写好的切片合并,一般是在store文件夹下新建一个index.js文件来合并切片,这里主要使用的是configureStore,代码如下:

import { configureStore } from '@reduxjs/toolkit';
import User from './user';export default configureStore({reducer: {user: User.reducer,// 有多少切片就一个个写进来即可}
})

(3)配置store和使用store

 (1)配置

在index.jsx文件或者main.jsx文件进行配置

先添加以下代码,将上面的index.js文件导入进去:

import store from './store';

然后用Provider组件包裹住<APP />,并且给属性绑定store,示例如下:

const root = ReactDOM.createRoot(document.getElementById('root'));
root.render(// <React.StrictMode><Provider store={store}><App /></Provider>// </React.StrictMode>
);

(2)使用

在需要使用状态的组件中进行编写代码,用到useSelector来拿数据,useDispatch来调用方法

import React from "react";
import { useSelector, useDispatch } from "react-redux";const PageOne = () => {const user = useSelector((state) => state.user);const dispatch = useDispatch();return (<div><h1>Page One</h1><div>{user.age}</div><button onClick={() => {dispatch({ type: "user/setAge" , payload: user.age + 1});}}>年龄+1</button></div>)
}export default PageOne;

但是这样的话这个 dispatch({ type: "user/setAge" , payload: user.age + 1})不够优雅,我们可以换一个优雅体面的方式

先将user引入组件中,这样就可以更方便的使用了,跟前面解释export const { setName, setAge } = useSlice.actions的写法的原因给callback了

import React from "react";
import { useSelector, useDispatch } from "react-redux";
import userSlice from "../../stores/user"; // 引入userSliceconst { setName, setAge } = userSlice.actions; // 解构出actions中的方法const PageOne = () => {const user = useSelector((state) => state.user);const dispatch = useDispatch();return (<div><h1>Page One</h1><div>{user.age}</div><button onClick={() => {dispatch(setName("李四"));dispatch(setAge(user.age + 1));}}>年龄+1, 改名李四</button></div>)
}export default PageOne;

好了,toolkit的用法基本就是这样,学会了就可以在编程中一边练一边不断学习更高阶的用法了,冲冲冲

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

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

相关文章

父组件prop传向子组件的值,被子组件直接v-model绑定 功能不生效

隐式修改组件属性会导致功能异常 实际操作中发现&#xff0c;即便是父组件把简单数据通过prop传给了子组件&#xff0c;子组件再使用v-model绑定&#xff0c;也不行&#xff0c;响应式还是对异常 原vue2业务中存在组件定义某个类型为Object的属性&#xff0c;然后将该属性对象…

c#bitconverter操作,不同变量类型转byte数组

缘起:串口数据传输的基础是byte数组&#xff0c;write(buff,0,num)或者writeline(string)&#xff0c;如果是字符串传输就是string变量就可以了&#xff0c;但是在modbus这类hex传递时&#xff0c;就要遇到转换了&#xff0c;拼凑byte数组时需要各种变量的值传递&#xff0c;解…

【Redis】set 类型

set 一. set 类型介绍二. set 命令sadd、smembers、sismemberscard、spop、srandmembersmove、srem集合间操作交集&#xff1a;sinter、sinterstore并集&#xff1a;sunion、sunionstore差集&#xff1a;sdiff、sdiffstore 三. set 命令小结四. set 内部编码方式五. set 使用场…

02-Redis常见命令

02-Redis常见命令 Redis数据结构介绍 Redis是一个key-value的数据库&#xff0c;key一般是String类型&#xff0c;不过value的类型多种多样&#xff1a; 贴心小建议&#xff1a;命令不要死记&#xff0c;学会查询就好啦 Redis为了方便学习&#xff0c;将操作不同数据类型的命…

Rk3568驱动开发_GPIO点亮LED_12

需求&#xff1a; 用配置寄存器方式控制点灯非常原始&#xff0c;现在采用更方便的Linux提供的pctrl和gpio子系统编写字符驱动 1.设备树配置&#xff1a; 现将开发板中呼吸灯关闭掉防止占用到我需要使用的引脚 /* Narnat 2025-5-29 RK3568 GPIO 无需设置pinctrl*/gpioled{co…

阿里云ACP云计算备考笔记 (3)——云存储RDS

目录 第一章 云存储概览 1、云存储通用知识 ① 发展历史 ② 云存储的优势 2、云存储分类 3、文件存储业务场景 第二章 块存储 1、块存储分类 2、云盘的优势 3、创建云盘 4、管理数据盘 ① 格式化数据盘 ② 挂载数据盘 ③ 通过 API 挂载云盘 5、管理系统盘 ① 更…

亚矩阵云手机实测体验:稳定流畅背后的技术逻辑​

最近在测试一款云手机服务时&#xff0c;发现亚矩阵的表现出乎意料地稳定。作为一个经常需要多设备协作的开发者&#xff0c;我对云手机的性能、延迟和稳定性要求比较高。经过一段时间的体验&#xff0c;分享一下真实感受&#xff0c;避免大家踩坑。 ​​1. 云手机能解决什么问…

STM32H562----------ADC外设详解

1、ADC 简介 STM32H5xx 系列有 2 个 ADC,都可以独立工作,其中 ADC1 和 ADC2 还可以组成双模式(提高采样率)。每个 ADC 最多可以有 20 个复用通道。这些 ADC 外设与 AHB 总线相连。 STM32H5xx 的 ADC 模块主要有如下几个特性: 1、可配置 12 位、10 位、8 位、6 位分辨率,…

【Android】双指旋转手势

一&#xff0c;概述 本文参考android.view.ScaleGestureDetector&#xff0c;对双指旋转手势做了一层封装&#xff0c;采用了向量计算法简单实现&#xff0c;笔者在此分享下。 二&#xff0c;实例 如下&#xff0c;使用RotateGestureDetector即可委托&#xff0c;实现旋转手…

B站的视频怎么下载下来——Best Video下载器

B站&#xff08;哔哩哔哩&#xff09;作为国内最受欢迎的视频平台之一&#xff0c;聚集了无数优质内容&#xff1a;动漫番剧、游戏实况、学习课程、纪录片、Vlog、鬼畜剪辑……总有那么些视频让人想反复观看、离线观看&#xff0c;甚至剪辑创作。 但你是否遇到过这样的烦恼&am…

基于SFC的windows系统损坏修复程序

前言 在平时使用Windows操作系统时会遇到很多因为系统文件损坏而出现的错误 例如:系统应用无法打开 系统窗口(例如开始菜单)无法使用 电脑蓝屏或者卡死 是如果想要修复很多人只能想到重装系统。但其实Windows有一个内置的系统文件检查器可以修复此类错误。 原理 SFC命令…

智绅科技 —— 智慧养老 + 数字健康,构筑银发时代安全防护网

在老龄化率突破 21.3% 的当下&#xff0c;智绅科技以 "科技适老" 为核心理念&#xff0c;构建 "监测 - 预警 - 干预 - 照护" 的智慧养老闭环。 其自主研发的七彩喜智慧康养平台&#xff0c;通过物联网、AI 和边缘计算技术&#xff0c;实现对老年人健康与安…

用函数实现模块化程序设计(适合考研、专升本)

函数 定义&#xff1a;本质上是一段可以被连续调用、功能相对独立的程序段 c语言是通过“函数”实现模块化的。根据分类标准不同函数分为以下几类。 用户角度&#xff1a;库函数、自定义函数 函数形式&#xff1a;有参函数、无参函数 作用域&#xff1a;外部函数、内部函数 …

OpenCV 滑动条调整图像亮度

一、知识点 1、int createTrackbar(const String & trackbarname, const String & winname, int * value, int count, TrackbarCallback onChange 0, void * userdata 0); (1)、创建一个滑动条并将其附在指定窗口上。 (2)、参数说明: trackbarname: 创建的…

vcs仿真产生fsdb波形的两种方式

目录 方法一&#xff1a; 使用verilog自带的系统函数 方法二&#xff1a; 使用UCLI command 2.1 需要了解什么是vcs的ucli&#xff0c;怎么使用ucli&#xff1f; 2.2 使用ucli dump波形的方法 使用vcs仿真产生fsdb波形有两种方式&#xff0c;本文参考《vcs user guide 20…

【前端】每日一道面试题6:解释Promise.any和Promise.allSettled的使用场景及区别。

Promise.any() 和 Promise.allSettled() 是 JavaScript 中用于处理异步操作的两种不同策略的 Promise 组合器&#xff0c;它们的核心区别在于逻辑目标与结果处理方式&#xff1a; 1. Promise.any() 使用场景&#xff1a; 需要获取 首个成功结果&#xff08;类似竞速成功优先&…

数据链路层__

文章目录 数据链路层基本概念&#xff08;1&#xff09;链路管理&#xff1a;面向连接的服务&#xff08;2&#xff09;帧同步&#xff1a;成帧1、字符计数法2、字符填充法&#xff08;带填充的首尾界符法&#xff09;3、带填充的首位标志法4、物理层编码违例法 &#xff08;3&…

coze智能体后端接入问题:

是否一定要按照coze官方API文档格式调用&#xff1f; 不一定&#xff1a;以下面代码为例(给了注释) app.route(/compare_models, methods[POST]) def compare_models():print("收到 compare_models 请求&#xff01;") #begin-这一部分代码作用&#xff1a;从前端接…

如何轻松、安全地管理密码(新手指南)

很多人会为所有账户使用相同、易记的密码&#xff0c;而且常常多年不换。虽然这样方便记忆&#xff0c;但安全性非常低。 您可能听说过一些大型网站的信息泄露事件&#xff0c;同样的风险也可能存在于您的WordPress网站中。如果有不法分子获取了访问权限&#xff0c;您的网站和…

宝塔think PHP8 安装使用FFmpeg 视频上传

宝塔think PHP8 安装使用FFmpeg 一、 安装think PHP8二、安装 FFmpeg1&#xff0c;登录到宝塔面板。2&#xff0c;进入“软件商店”。3&#xff0c;搜索“FFmpeg”。4&#xff0c;选择版本点击安装。5&#xff0c;检查 FFmpeg 是否安装成功6&#xff0c; 在 ThinkPHP 8 中使用 …