React-Redux 中实现数据更新操作的正确姿势

React-Redux 中实现数据更新操作的正确姿势

本文深入探讨了在 react-redux 应用中实现数据更新功能时常见的错误及其解决方案。通过分析一个联系人管理应用的案例,我们详细解释了动作创建器(action creator)与 reducer 之间有效载荷(payload)不匹配的问题,并提供了修正后的代码示例,确保数据更新逻辑的正确性和一致性,帮助开发者避免类似陷阱。

在 React-Redux 应用程序中,管理数据状态的更新是一个核心功能。虽然添加和删除操作通常较为直接,但实现数据编辑(更新)功能时,开发者常会遇到因动作创建器与 Reducer 之间有效载荷(Payload)不一致而导致的问题。本教程将以一个联系人管理应用为例,详细解析这一常见陷阱,并提供清晰的解决方案。

理解 React-Redux 更新流程

一个典型的 React-Redux 数据更新流程包括以下几个步骤:

用户交互触发更新: 用户在 UI 上点击“更新”按钮或提交表单。组件调度动作: React 组件通过 useDispatch 钩子调度一个更新动作。动作创建器生成动作: 动作创建器负责构造一个包含 type 和 payload 的动作对象。Reducer 处理动作: Reducer 接收到动作后,根据 type 识别操作类型,并利用 payload 中的数据生成新的状态。状态更新与 UI 重新渲染: Redux Store 更新状态,React 组件检测到状态变化并重新渲染。

发现问题:动作创建器与 Reducer 的有效载荷不匹配

在提供的联系人应用示例中,更新功能的核心问题在于 UpdateContact 动作创建器和 AppReducer 中 UPDATE_CONTACT 对应的处理逻辑之间存在有效载荷的期望不一致。

首先,我们来看 UpdateContact 动作创建器的定义:

// actions/Actions.jsexport const UpdateContact = (id) => { // 注意:这里只接收了 id    return {        type: 'UPDATE_CONTACT',        payload: id // 有效载荷是联系人的 id    }}

从上述代码可以看出,UpdateContact 动作创建器被设计为接收一个 id 作为参数,并将其作为动作的 payload。这意味着当这个动作被调度时,Reducer 将会收到一个 payload 为联系人 id 的动作对象。

然而,AppReducer 中 UPDATE_CONTACT 类型的处理逻辑却期望 payload 是一个完整的联系人对象,用于替换旧的联系人:

// reducers/AppReducer.jsexport const AppReducer = (state = initialState, action) => {    switch (action.type) {        // ... 其他 case        case "UPDATE_CONTACT":            const updatedContact = action.payload; // 这里期望 payload 是一个完整的联系人对象            const updatedContacts = state.contacts.map((contact) => {                if (contact.id === updatedContact.id) {                    return updatedContact // 用新的联系人对象替换旧的                }                return contact            })            return updatedContacts // 注意:这里直接返回了 updatedContacts,这本身也是一个潜在问题,应返回新的状态对象        // ... 其他 case    }}

当 UpdateContact 动作被调度时,如果 payload 仅仅是一个 id(例如 payload: “some-id”),那么 updatedContact 变量将直接被赋值为这个 id。随后,在 map 循环中,contact.id === updatedContact.id 的比较将是 contact.id === “some-id”。即使找到了匹配项,return updatedContact 也将返回这个 id,而不是一个完整的更新后的联系人对象。这显然不是我们期望的更新行为。

解决方案:统一有效载荷的期望

要解决这个问题,我们需要确保动作创建器发送的 payload 与 Reducer 期望接收的 payload 类型和结构完全一致。在这种情况下,Reducer 期望接收一个完整的、已更新的联系人对象。

腾讯智影-AI数字人 腾讯智影-AI数字人

基于AI数字人能力,实现7*24小时AI数字人直播带货,低成本实现直播业务快速增增,全天智能在线直播

腾讯智影-AI数字人 73 查看详情 腾讯智影-AI数字人

步骤一:修改 UpdateContact 动作创建器

将 UpdateContact 动作创建器修改为接收一个完整的联系人对象作为参数,并将其作为 payload:

// actions/Actions.js (修正后)export const UpdateContact = (contact) => { // 现在接收的是一个完整的 contact 对象    return {        type: 'UPDATE_CONTACT',        payload: contact // 有效载荷是完整的联系人对象    }}

步骤二:修改 UpdateContactPage 组件中的调度逻辑

在 UpdateContactPage 组件中,当用户提交更新表单时,我们应该调度包含完整 user(即更新后的联系人)对象的动作:

// components/UpdateContactPage.jsx (修正后)import React, { useState, useEffect } from 'react'; // 引入 useEffectimport { useSelector, useDispatch } from 'react-redux';import { useParams, useNavigate } from 'react-router-dom'; // 引入 useNavigateimport { UpdateContact } from '../redux/actions/Actions';const UpdateContactPage = () => {  const { id } = useParams();  const contacts = useSelector(state => state.userReducer.contacts);  const navigate = useNavigate(); // 用于导航  // 确保在组件挂载时找到对应的联系人并初始化状态  const initialContact = contacts.find((contact) => contact.id === id);  // 如果找不到联系人,可以重定向或显示错误  useEffect(() => {    if (!initialContact) {      navigate('/contacts'); // 或者显示错误信息    }  }, [initialContact, navigate]);  // 使用 initialContact 初始化 user 状态  const [user, setUser] = useState(initialContact || {    id: '',    userName: '',    surname: '',    image: ''  });  const handleChange = (e) => {    setUser({ ...user, [e.target.name]: e.target.value });  };  const dispatch = useDispatch();  const updateContactForm = (e) => {    e.preventDefault();    dispatch(UpdateContact(user)); // 现在调度的是包含完整 user 对象的动作    navigate('/contacts'); // 更新后导航回联系人列表页面  };  if (!initialContact) {    return 
Contact not found.
; // 或者加载指示器 } return (
{/* 使用 value 绑定 */}
{/* 使用 value 绑定 */}
{/* 使用 value 绑定 */}

重要改进点:

useState 初始化: 直接使用 initialContact 初始化 user 状态,确保表单显示的是当前联系人的最新数据。受控组件: 将 defaultValue 改为 value,并结合 onChange 实现受控组件,这是 React 表单的最佳实践。找不到联系人的处理: 增加了当 id 对应的联系人不存在时的处理逻辑,例如重定向。导航: 引入 useNavigate 在更新成功后进行页面跳转。

步骤三:验证 Reducer 逻辑

在修改了动作创建器和调度逻辑后,AppReducer 中的 UPDATE_CONTACT 逻辑现在是正确的,因为它会接收到一个完整的 updatedContact 对象,并用它来替换 contacts 数组中匹配 id 的旧对象。

// reducers/AppReducer.js (验证后,并修正了返回状态的方式)export const AppReducer = (state = initialState, action) => {    switch (action.type) {        // ... 其他 case        case "UPDATE_CONTACT":            const updatedContact = action.payload; // 现在 action.payload 确实是完整的联系人对象            const updatedContacts = state.contacts.map((contact) => {                if (contact.id === updatedContact.id) {                    return updatedContact; // 返回更新后的联系人对象                }                return contact;            });            return { // Reducer 必须返回一个新的状态对象                ...state,                contacts: updatedContacts            };        // ... 其他 case    }}

Reducer 修正点:

Reducers 必须返回一个新的状态对象,而不是直接返回一个数组。原始代码中的 return updatedContacts 是一个错误,因为它改变了 Redux Store 状态的结构。正确的做法是返回一个包含所有原有状态属性(通过 ...state 展开)和更新后的 contacts 数组的新状态对象。

总结与最佳实践

有效载荷一致性: 确保动作创建器发送的 payload 结构与 Reducer 期望接收的结构完全匹配。这是避免 Redux 更新逻辑错误的关键。Reducer 纯粹性与不变性: Reducer 必须是纯函数,不应有副作用。它应该始终返回一个新的状态对象,而不是直接修改现有状态。对于数组和对象,这意味着要创建新的数组或对象副本。受控组件: 在 React 中处理表单输入时,优先使用受控组件,通过 value 和 onChange 绑定表单元素的值到组件状态。调试工具 利用 Redux DevTools 可以清晰地查看每个动作的 type 和 payload,以及状态的变化,这对于调试此类问题非常有帮助。

通过遵循这些原则,您可以更有效地在 React-Redux 应用程序中实现健壮的数据更新功能。

以上就是React-Redux 中实现数据更新操作的正确姿势的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/297449.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月4日 23:28:40
下一篇 2025年11月4日 23:29:36

相关推荐

  • 干净的架构:从哪里开始?

    在上一篇文章中我们有: 我们的问题域:具有一些要求的 todo 应用程序配置为使用 python 和 python polylith 的基本存储库。 因此,一些决定已经完成。我们拥有一些工具并已经决定了存储库的外观。 这是我喜欢 polylith 的原因之一:无论您编码什么或您的组织有多大,所有存储…

    2025年12月13日
    000
  • AoC &#- Day 仔细考虑(C# 和 Python)

    仔细考虑一下 今天的挑战我第一次看到regex时就尖叫起来,主要是因为每当我看到“提取该字符串的一部分”时,regex就是我的首选; 基本概念和要求 所以我们需要找到所有 mul(number1, number2) 并将它们相乘,但忽略所有其他字符。 所以我们需要找到一种机制来查找所有有效的 mul…

    2025年12月13日
    000
  • 打包 Python RPM

    最近,我正在为 Red Hat 工作的当前项目中执行一项非常具体的任务,即 RHEL LightspeedShellAI,这个项目相对较新,但我们想开始发货开发 RPM,让我们的 QE 朋友开始使用该工具并在他们的管道中进行测试。 我知道包装和一般Python东西的方法,但是伙计,我必须告诉你,这个…

    2025年12月13日
    000
  • 无法从 Companies House 和 HMRC API 获取营业额数据超过英镑的数据

    无法从 Companies House 和 HMRC API 获取营业额数据超过 200 万英镑的数据 2024 年 12 月 4 日 评论:2 答案:0 0 我已经处理这个问题一天了,但在从 hmrc 和 companies house api 获取公司数据时遇到了麻烦。我主要需要营业额超过 10…

    2025年12月13日
    000
  • python中grid的用法

    Python 中的 Tkinter 库中的网格小部件用于创建带有规则排列行和列的网格状布局,以便组织和对齐图形元素。要使用网格,可以使用以下步骤:使用 Grid 类创建网格。使用 rowconfigure() 和 columnconfigure() 方法配置行和列的属性。使用 grid() 方法将小…

    2025年12月13日
    000
  • 如何使用 Python 抓取 Google 搜索结果

    抓取 google 搜索可提供基本的 serp 分析、seo 优化和数据收集功能。现代抓取工具使这个过程更快、更可靠。 我们的一位社区成员撰写了此博客,作为对 crawlee 博客的贡献。如果您想向 crawlee 博客贡献此类博客,请通过我们的 discord 频道与我们联系。 在本指南中,我们将…

    2025年12月13日 好文分享
    000
  • python爬虫怎么点按钮

    使用 Seleniumwebdriver 来模拟浏览器操作,通过单击元素的 ID 或 XPath 来点击按钮。步骤包括:安装 Seleniumwebdriver、导入模块、创建 WebDriver 实例、导航到页面、定位按钮(使用 find_element_by_id() 或 find_elemen…

    2025年12月13日
    000
  • 快速编程

    快速编程是指自动插入代码并解决简单问题的模板。 从文件中卸载样本并用标签填充数据数组是可以复制并粘贴到不同项目中的基本操作。 from google.colab import drivedrive.mount(‘/content/gdrive’, force_remount=true)!cp /co…

    2025年12月13日
    000
  • [CVHSV 与 RGB:理解和利用 HSV 进行图像处理

    在上一篇文章中,我们探索了在 opencv 中处理 rgb 图像的基础知识,包括绘图以及调整亮度和对比度。虽然 rgb 色彩空间非常适合计算机显示器,因为它以屏幕发出的光强度来表示颜色,但它与人类在自然世界中感知颜色的方式并不相符。这就是 hsv(色相、饱和度、明度)发挥作用的地方——一个旨在以更接…

    2025年12月13日 好文分享
    000
  • 人工智能如何塑造更智能的游戏和模拟世界

    人工智能 (AI) 正在迅速改变游戏格局,使视频游戏变得更加身临其境、智能且引人入胜。从增强 NPC(非玩家角色)行为到创建逼真的虚拟环境,人工智能是游戏开发和模拟体验中许多创新的核心。 在本文中,我们将探讨人工智能如何将游戏提升到一个新的水平,深入探讨埃隆·马斯克对游戏中人工智能的愿景,并讨论这些…

    2025年12月13日
    000
  • 理解自我反思的简单代码(代理设计模式)

    反思/自我反思有点被低估了。如果您的应用程序依赖于提示,我强烈建议您探索这个概念。实施起来并不难,反思技术可以帮助迭代地完善 llm 响应。 from mirascope.core import BaseMessageParam, ResponseModelConfigDict, openaifro…

    2025年12月13日
    000
  • Python 初学者指南:快速教程 – 2

    python 是最流行的编程语言之一,以其简单性和多功能性而闻名。无论您是编程新手还是希望为您的项目选择 python,本教程都将指导您完成基础知识。 1.什么是python? python 是一种高级解释型编程语言,强调可读性和效率。它广泛应用于网页开发、数据分析、人工智能、科学计算等领域。 2.…

    2025年12月13日
    000
  • 为您的 Python 项目设置 Conda 环境 – 1

    使用 conda 和requirements.txt 设置 python 项目 在处理 python 项目时,创建隔离的环境来管理依赖关系并避免冲突至关重要。本指南将帮助您安装 anaconda、解决常见问题并为您的项目设置虚拟环境。 1.安装 anaconda(在根终端) a) 按照本指南安装 a…

    2025年12月13日
    000
  • python做爬虫 怎么样效率最高

    高效爬虫使用 Python 的优化策略:并行化处理:使用多线程或多进程同时处理多个请求,并利用 asyncio 或 Tornado 实现非阻塞 I/O。缓存请求:存储爬取结果以避免重复爬取。限制请求速率:使用速率限制器控制爬取频率,避免触发网站反爬机制。针对性爬取:使用正则表达式或 CSS 选择器只…

    2025年12月13日
    000
  • python爬虫怎么爬取动态页面

    Python 爬取动态页面的方法包括:Selenium:用于自动化 Web 浏览器,适合复杂的动态页面。Splash:基于 Chromium headless browser,通过 JavaScript API 处理动态页面。Puppeteer:无头 Chrome 浏览器,提供强大 API 控制浏览…

    2025年12月13日
    000
  • python爬虫怎么模拟点击

    要使用 Python 爬虫模拟点击,可以使用以下方法:使用 Selenium,找到要点击的元素并执行 click 方法。使用 Headless 浏览器,如 Puppeteer 或 Playwright,使用 JavaScript 或 Python 代码控制无头浏览器,并执行点击操作。 如何使用 Py…

    2025年12月13日
    000
  • 给一个关键词用python怎么爬虫

    使用 Python 进行网络爬虫步骤:安装 beautifulsoup4、requests 和 lxml 库。发送 HTTP GET 请求到目标 URL。解析 HTML 响应并提取数据。处理并存储提取的数据。 使用 Python 进行网络爬虫 网络爬虫是一种自动化程序,用于从互联网上提取数据。它通过…

    2025年12月13日
    000
  • python爬虫怎么处理弹窗

    Python 爬虫可以采用多种方法处理弹窗:模态弹窗(覆盖整个浏览器窗口):使用 Selenium 或 PyAutoGUI 找到并关闭弹窗。非模态弹窗(不阻止页面交互):等待弹窗加载,使用 Selenium 或 PyAutoGUI 查找并关闭弹窗。 Python 爬虫如何处理弹窗 在使用 Pytho…

    2025年12月13日
    000
  • 使用 LlamaChat 和 Excel 构建一个简单的聊天机器人]

    在这篇文章中,我将解释如何使用 llama2 模型构建一个聊天机器人来智能查询 excel 数据。 我们正在建设什么 加载 excel 文件。将数据分割成可管理的块。将数据存储在矢量数据库中以便快速检索。使用本地 llama2 模型来回答基于 excel 文件的内容。 先决条件: python (≥…

    2025年12月13日
    000
  • DSPy:语言模型编程的新方法

    挑战:超越传统提示 在使用语言模型 (llm) 时,开发人员面临着一系列常见的挑战。我们花费了无数的时间来制作完美的提示,却发现当我们切换模型或输入略有变化时,我们精心设计的解决方案就会崩溃。传统的即时工程方法是手动的、耗时的,而且通常是不可预测的。 解决方案:斯坦福大学的 dspy 框架 dspy…

    2025年12月13日
    000

发表回复

登录后才能评论
关注微信