React状态管理Context API + useReducer

news2025/6/6 11:36:21

在 React 中,Context API + useReducer 是一种轻量级的状态管理方案,适合中小型应用或需要跨组件共享复杂状态的场景。它避免了 Redux 的繁琐配置,同时提供了清晰的状态更新逻辑。


1. 基本使用步骤

(1) 定义 Reducer

类似于 Redux 的 reducer,用于处理状态更新逻辑:

// reducer.js
export const initialState = {
  count: 0,
  user: null,
};

export function reducer(state, action) {
  switch (action.type) {
    case 'INCREMENT':
      return { ...state, count: state.count + 1 };
    case 'DECREMENT':
      return { ...state, count: state.count - 1 };
    case 'SET_USER':
      return { ...state, user: action.payload };
    default:
      return state;
  }
}

(2) 创建 Context 和 Provider

使用 createContext 创建 Context,并用 useReducer 管理状态:

// AppContext.js
import { createContext, useReducer } from 'react';
import { initialState, reducer } from './reducer';

// 1. 创建 Context
export const AppContext = createContext();

// 2. 创建 Provider 组件
export function AppProvider({ children }) {
  const [state, dispatch] = useReducer(reducer, initialState);

  return (
    <AppContext.Provider value={{ state, dispatch }}>
      {children}
    </AppContext.Provider>
  );
}

(3) 在顶层组件包裹 Provider

// App.js
import { AppProvider } from './AppContext';
import Counter from './Counter';

function App() {
  return (
    <AppProvider>
      <Counter />
    </AppProvider>
  );
}

(4) 在子组件使用状态

通过 useContext 获取 statedispatch

// Counter.js
import { useContext } from 'react';
import { AppContext } from './AppContext';

export function Counter() {
  const { state, dispatch } = useContext(AppContext);

  return (
    <div>
      <p>Count: {state.count}</p>
      <button onClick={() => dispatch({ type: 'INCREMENT' })}>+</button>
      <button onClick={() => dispatch({ type: 'DECREMENT' })}>-</button>
    </div>
  );
}

2. 数据持久化

Context + useReducer 管理的状态是纯内存状态,当页面刷新时,这些数据会丢失,因为 JavaScript 的内存会被清空,恢复到初始状态。
在 React 的 Context + useReducer 架构中实现数据持久化( localStorage sessionStorage,页面刷新不丢失)

2.1实现步骤

(1)定义 Reducer(持久化/非持久化/临时数据)
// reducer.js
export const initialState = {
  // 需要持久化的数据
  persistedData: { language: 'en' },
  // 非持久化数据
  sessionData: { loginForm: null },
  // 临时数据
  tempData: {
    count: 0
  }
};

export function reducer(state, action) {
  switch (action.type) {
  	// 持久化数据
     case 'SET_LANGUAGE':
      return { 
        ...state, 
        persistedData: { ...state.persistedData, language: action.payload } 
      };
     // 非持久化数据
    case 'SET_LOGIN_FORM':
      return { 
        ...state, 
        sessionData: { ...state.sessionData, loginForm: action.payload } 
      };
      // 临时数据
    case 'SET_COUNT':
      return {
        ...state, 
        tempData: { ...state.tempData, count: action.payload } 
      }
    default:
      return state;
  }
}
(2)带持久化的Provider组件实现
// AppContext.jsx
import { createContext, useReducer, useEffect } from 'react';
import { initialState, reducer } from './reducer';

// 1. 创建 Context
export const AppContext = createContext();

// 2. 创建 Provider 组件
export function AppProvider({ children }) {
  const [state, dispatch] = useReducer(
    reducer,
    {
      ...initialState,
      persistedData: JSON.parse(localStorage.getItem('persistedData')) || initialState.persistedData,
      sessionData: JSON.parse(sessionStorage.getItem('sessionData')) || initialState.sessionData
    }
  );
  
  // 监听localStorage字段的变化
  useEffect(() => {
    localStorage.setItem('persistedData', JSON.stringify(state.persistedData));
  }, [state.persistedData]);
  
  // 监听sessionStorage 字段的变化
  useEffect(() => {
    sessionStorage.setItem('sessionData', JSON.stringify(state.sessionData));
  }, [state.sessionData]);
  return (
    <AppContext.Provider value={{ state, dispatch }}>
      {children}
    </AppContext.Provider>
  );
}
(3)在顶层组件包裹 Provider
// app.jsx
import { RouterProvider } from "react-router";
import router from "./router/index.jsx";
import { AppProvider } from '@stores/AppContext.jsx';
function App() {
  return (
    <AppProvider>
    <div className="app">
      <RouterProvider router={router} />
    </div>
    </AppProvider>
  )
}

export default App

2.2 纯 localStorage 与 Context + useReducer + localStorage对比

1、 纯 localStorage特点

优点缺点
1. 实现简单,无需额外库1. 状态不同步:多个组件无法实时共享同一份数据
2. 适合极简场景2. 重复代码:每个组件需单独处理存储逻辑
3. 无性能开销(仅读写存储)3. 难以维护:业务复杂时逻辑分散

2、 Context + useReducer + localStorage特点

优点缺点
1. 状态全局共享:所有组件实时响应变化1. 代码量稍多(需设置 Context/Reducer)
2. 逻辑集中:易于维护和扩展2. 小型项目可能过度设计
3. 自动持久化:状态变更自动同步到存储3. 需处理 Provider 嵌套问题

核心区别对比

对比维度纯 localStorageContext + useReducer + localStorage
状态同步需手动触发,组件间不同步自动同步,全局状态一致
代码组织逻辑分散在各组件集中管理,高内聚低耦合
维护性难扩展,易出现重复代码易于扩展和维护
性能直接操作存储,无额外开销有 Context 的渲染开销(可通过 memo 优化)
适用场景简单页面、独立组件中大型应用、需共享状态的场景

总结

  • 直接 localStorage:简单粗暴,适合局部状态。
  • Context + useReducer + localStorage:专业方案,适合全局状态。

2.3注意事项

  • localStorage /sessionStorage只能存字符串,复杂数据需用 JSON.stringify
  • 优点:1、代码简洁,适合小型应用;2、无需第三方库;
  • 缺点: 1、 localStorage/sessionStorage是同步操作,可能阻塞主线程;2、存储大小有限(通常 5MB);

2. 进阶优化

(1) 封装自定义 Hook

避免在每个组件里重复写 useContext

// hooks/useAppContext.js
import { useContext } from 'react';
import { AppContext } from '../AppContext';

export function useAppContext() {
  return useContext(AppContext);
}

然后在组件中使用:

const { state, dispatch } = useAppContext();

(2) 优化性能(避免不必要的渲染)

默认情况下,Context 的更新会导致所有消费者组件重新渲染。可以使用 memo + 拆分 Context 优化:

// 拆分多个 Context
const CountContext = createContext();
const UserContext = createContext();

// 在 Provider 里分别提供
<CountContext.Provider value={{ countState, countDispatch }}>
  <UserContext.Provider value={{ userState, userDispatch }}>
    {children}
  </UserContext.Provider>
</CountContext.Provider>

(3) 结合异步操作

可以在 dispatch 里处理异步逻辑(如 API 请求):

async function fetchUser(dispatch) {
  try {
    const user = await fetch('/api/user').then(res => res.json());
    dispatch({ type: 'SET_USER', payload: user });
  } catch (error) {
    dispatch({ type: 'SET_ERROR', payload: error.message });
  }
}

// 在组件中调用
fetchUser(dispatch);

3. 优缺点对比

优点缺点
无需额外库,React 原生支持大型应用可能性能较差(需手动优化)
比 Redux 更轻量异步处理较麻烦(需手动封装)
适合中小型应用调试不如 Redux 方便(无 DevTools)

4. 适用场景

  • 小型/中型应用:不想引入 Redux 或 Zustand 时。
  • 组件层级较深:需要跨多层传递状态时。
  • 简单全局状态:如主题、用户登录信息等。

总结

  • Context + useReducer 是 React 内置的状态管理方案,适合轻量级需求。
  • 对于更复杂的状态管理,可考虑 ZustandRedux Toolkit
  • 如果涉及大量异步逻辑,建议结合 React QuerySWR 使用。

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

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

相关文章

Doris Catalog 联邦分析查询性能优化:从排查到优化的完整指南

在大数据分析中&#xff0c;Doris 的 Catalog 联邦分析功能为整合多源数据提供了有力支持。然而&#xff0c;在实际应用中&#xff0c;可能会遇到各种问题影响其正常运行。本文将详细剖析这些问题并提供解决方案。 一、联邦分析查询慢&#xff1a;内外表通用排查逻辑 当遇到 …

01 Deep learning神经网络的编程基础 二分类--吴恩达

二分类 1. 核心定义 二分类任务是监督学习中最基础的问题类型&#xff0c;其目标是将样本划分为两个互斥类别。设样本特征空间为 X ⊆ R n \mathcal{X} \subseteq \mathbb{R}^n X⊆Rn&#xff0c;输出空间为 Y { 0 , 1 } \mathcal{Y} \{0,1\} Y{0,1}&#xff0c;学习目标为…

视频自动化分割方案:支持按时间与段数拆分

在日常视频处理任务中&#xff0c;如何快速将一个较长的视频文件按照指定规则拆分为多个片段&#xff0c;是许多用户都会遇到的问题。尤其对于需要批量处理视频的开发者、自媒体运营者或内容创作者来说&#xff0c;手动剪辑不仅效率低下&#xff0c;还容易出错。这是一款绿色免…

股指期货合约价值怎么算?

股指期货合约价值就是你买一手股指期货合约&#xff0c;理论上值多少钱。这个价值是根据期货的价格和合约乘数来计算的。就好比你买了一斤苹果&#xff0c;价格是5块钱一斤&#xff0c;那你买一斤就得付5块钱。股指期货也是一样&#xff0c;只不过它的计算稍微复杂一点点。 一…

【QT】使用QT帮助手册找控件样式

选择帮助—》输入stylesheet(小写)—》选择stylesheet—》右侧选择Qt Style Sheets Reference 2.使用CtrlF—》输入要搜索的控件—》点击Customizing QScrollBar 3.显示参考样式表–》即可放入QT-designer的样式表中

计算机网络(5)——数据链路层

1.概述 数据链路层负责一套链路上从一个节点向另一个物理链路直接相连的相邻节点传输数据报。换言之&#xff0c;主要解决相邻节点间的可靠数据传输 节点(nodes)&#xff1a;路由器和主机 链路(links)&#xff1a;连接相邻节点的通信信道 2.数据链路层服务 2.1 组帧 组帧(fra…

VuePress完美整合Toast消息提示

VuePress 整合 Vue-Toastification 插件笔记 记录如何在 VuePress 项目中整合使用 vue-toastification 插件&#xff0c;实现优雅的消息提示。 一、安装依赖 npm install vue-toastification或者使用 yarn&#xff1a; yarn add vue-toastification二、配置 VuePress 客户端增…

adb 连不上真机设备问题汇总

问题一、无法弹出 adb 调试授权弹窗 详细描述&#xff1a; 开发者选项中已打开 usb 调试&#xff0c;仅充电模式下 usb 调试也已打开&#xff0c;电脑通过 usb 连上手机后&#xff0c;一直弹出 adb 调试授权弹窗&#xff0c;尝试取消授权再次连接&#xff0c;还是无法弹出问题…

[yolov11改进系列]基于yolov11引入注意力机制SENetV1或者SENetV2的python源码+训练源码

本文给大家带来的改进机制是SENet&#xff08;Squeeze-and-Excitation Networks&#xff09;其是一种通过调整卷积网络中的通道关系来提升性能的网络结构。SENet并不是一个独立的网络模型Q&#xff0c;而是一个可以和现有的任何一个模型相结合的模块&#xff08;可以看作是一种…

鸿蒙仓颉语言开发实战教程:商城搜索页

大家好&#xff0c;今天要分享的是仓颉语言商城应用的搜索页。 搜索页的内容比较多&#xff0c;都有点密集恐惧症了&#xff0c;不过我们可以从上至下将它拆分开来&#xff0c;逐一击破。 导航栏 搜索页的的最顶部是导航栏&#xff0c;由返回按钮和搜索框两部分组成,比较简单…

图像去雾数据集总汇

自然去雾数据集 部分的数据清洗可以看这里&#xff1a;图像去雾数据集的下载和预处理操作 RESIDE-IN 将ITS作为训练集&#xff0c;SOTSindoor作为测试集。训练集13990对&#xff0c;验证集500对。 目前室内sota常用&#xff0c;最高已经卷到PSNR-42.72 最初应该是dehazefo…

网络攻防技术十四:入侵检测与网络欺骗

文章目录 一、入侵检测概述二、入侵系统的分类三、入侵检测的分析方法1、特征检测&#xff08;滥用检测、误用检测&#xff09;2、异常检测 四、Snort入侵检测系统五、网络欺诈技术1、蜜罐2、蜜网3、网络欺骗防御 六、简答题1. 入侵检测系统对防火墙的安全弥补作用主要体现在哪…

C++笔记-C++11(一)

1.C11的发展历史 C11 是 C 的第⼆个主要版本&#xff0c;并且是从 C98 起的最重要更新。它引⼊了⼤量更改&#xff0c;标准化了既有实践&#xff0c;并改进了对 C 程序员可⽤的抽象。在它最终由 ISO 在 2011 年 8 ⽉ 12 ⽇采纳前&#xff0c;⼈们曾使⽤名称“C0x”&#xff0c;…

JVM 类初始化和类加载 详解

类初始化和类加载 类加载的时机 加载、验证、准备、初始化和卸载这五个阶段的顺序是确定的&#xff0c;类型的加载过程必须按照这种顺序按部就班地开始&#xff0c;而解析阶段则不一定&#xff1a;它在某些情况下可以在初始化阶段之后再开始&#xff08;懒解析&#xff09;&am…

B站缓存视频数据m4s转mp4

B站缓存视频数据m4s转mp4 结构分析 结构分析 在没有改变数据存储目录的情况下&#xff0c;b站默认数据保存目录为&#xff1a; Android->data->tv.danmaku.bili->download每个文件夹代表一个集合的视频&#xff0c;比如&#xff0c;我下载的”java从入门到精通“&…

DeepSeek 助力 Vue3 开发:打造丝滑的日历(Calendar),日历_天气预报日历示例(CalendarView01_18)

前言&#xff1a;哈喽&#xff0c;大家好&#xff0c;今天给大家分享一篇文章&#xff01;并提供具体代码帮助大家深入理解&#xff0c;彻底掌握&#xff01;创作不易&#xff0c;如果能帮助到大家或者给大家一些灵感和启发&#xff0c;欢迎收藏关注哦 &#x1f495; 目录 Deep…

【机器学习】主成分分析 (PCA)

目录 一、基本概念 二、数学推导 2.1 问题设定&#xff1a;寻炸最大方差的投影方向 2.2 数据中心化 2.3 目标函数&#xff1a;最大化投影后的方差 2.4 约束条件 2.5 拉格朗日乘子法 ​编辑 2.6 主成分提取 2.7 降维公式 三、SVD 四、实际案例分析 一、基本概念 主…

二叉树-104.二叉树的最大深度-力扣(LeetCode)

一、题目解析 这里需要注意根节点的深度是1&#xff0c;也就是说计算深度的是从1开始计算的 二、算法原理 解法1&#xff1a;广度搜索&#xff0c;使用队列 解法2&#xff1a;深度搜索&#xff0c;使用递归 当计算出左子树的深度l&#xff0c;与右子树的深度r时&#xff0c;…

物料转运人形机器人适合应用于那些行业?解锁千行百业的智慧物流革命

当传统物流设备困于固定轨道&#xff0c;当人力搬运遭遇效率与安全的天花板&#xff0c;物料转运人形机器人正以颠覆性姿态重塑产业边界。富唯智能凭借GRID大模型驱动的"感知-决策-执行"闭环系统&#xff0c;让物料流转从机械输送升级为智慧调度——这不仅是工具的革…

时序预测模型测试总结

0.背景描述 公司最近需要在仿真平台上增加一些AI功能&#xff0c;针对于时序数据&#xff0c;想到的肯定是时序数据处理模型&#xff0c;典型的就两大类&#xff1a;LSTM 和 tranformer 。查阅文献&#xff0c;找到一篇中石化安全工程研究院有限公司的文章&#xff0c;题目为《…