React基础 第十七章(良好的State结构)

在React中,选择合适的state结构是构建高效、可维护应用程序的关键。本文将深入探讨如何根据React官方文档的建议来选择state结构,并提供实用的技巧、示例代码以及注意事项。

合并关联的State

当你发现自己在多个state变量上执行相同的操作时,最好将它们合并为一个单独的state变量。

技巧

  • 使用对象或数组来组合相关的state。
  • 当更新state时,确保更新所有相关字段。

示例

jsx 复制代码
const [position, setPosition] = useState({ x: 0, y: 0 });

function handlePointerMove(e) {
  setPosition({ x: e.clientX, y: e.clientY });
}

注意事项

  • 当使用对象作为state时,更新单个字段时不要忘记保留其他字段。
  • 或者使用immer

正确代码

jsx 复制代码
setPosition(prevPosition => ({ ...prevPosition, x: 100 }));

错误代码

jsx 复制代码
setPosition({ x: 100 }); // 错误:丢失了y属性

避免矛盾的State

确保state之间不会出现逻辑上的矛盾,例如,一个变量表示正在发送中,另一个变量表示已发送,这两个状态不应该同时出现。

技巧

  • 使用枚举或状态机来管理相关的状态。
  • 用一个state变量替代多个可能导致矛盾的state变量。

示例

jsx 复制代码
const [status, setStatus] = useState('typing'); // 'typing', 'sending', 'sent'

function handleSubmit(e) {
  e.preventDefault();
  setStatus('sending');
  sendMessage(text).then(() => setStatus('sent'));
}

注意事项

  • 避免创建逻辑上可能冲突的多个state。

正确代码

jsx 复制代码
setStatus('sending');

错误代码

jsx 复制代码
// 错误:定义多个状态,可能会导致状态冲突
setIsSending(true);
setIsSent(false);

避免冗余的State

不要在state中存储可以从props或其他state计算得出的值。

技巧

  • 使用计算属性或者在渲染方法中直接计算出这些值。

示例

jsx 复制代码
const [firstName, setFirstName] = useState('');
const [lastName, setLastName] = useState('');
const fullName = `${firstName} ${lastName}`;

注意事项

  • 不要将可计算的数据存储为state。

正确代码

jsx 复制代码
const fullName = `${firstName} ${lastName}`;

错误代码

jsx 复制代码
const [fullName, setFullName] = useState(''); // 错误:这是一个冗余的state

避免重复的State

当相同的数据在多个地方出现时,应该考虑将其集中管理。

技巧

  • 在state中存储唯一的标识符,而不是整个数据对象。
  • 使用查找表来管理数据。

示例

jsx 复制代码
const [items, setItems] = useState([{ id: 1, name: 'Item 1' }]);
const [selectedId, setSelectedId] = useState(1);
const selectedItem = items.find(item => item.id === selectedId);

注意事项

  • 避免在不同的state变量中存储相同的数据。

正确代码

jsx 复制代码
const selectedItem = items.find(item => item.id === selectedId);

错误代码

jsx 复制代码
const [selectedItem, setSelectedItem] = useState(items[0]); // 错误:创建了不必要的重复state

避免深度嵌套的State

深度嵌套的state很难管理和更新。尽可能地扁平化你的state结构。

技巧

  • 使用扁平化的数据结构,如ID映射表。
  • 将复杂的state逻辑拆分到更小的组件中。

示例

jsx 复制代码
const [plan, setPlan] = useState({
  0: { id: 0, title: 'Root', childIds: [1, 2] },
  1: { id: 1, title: 'Child 1', childIds: [] },
  2: { id: 2, title: 'Child 2', childIds: [] }
});

function handleDelete(id) {
  setPlan(prevPlan => {
    const newPlan = { ...prevPlan };
    delete newPlan[id];
    return newPlan;
  });
}

注意事项

  • 避免在state中使用过于复杂的数据结构。

正确代码

jsx 复制代码
const newPlan = { ...prevPlan };
delete newPlan[id];

错误代码

jsx 复制代码
setPlan(prevPlan => {
  prevPlan[id].childIds = []; // 错误:直接修改了state
  return prevPlan;
});

记住,简化你的state结构可以减少bug,提高性能,并使你的代码更易于理解。

相关推荐
NiceCloud喜云4 小时前
Opus 4.8 的 Effort Control 怎么选:Low 到 Max 五档策略
android·java·大数据·前端·c++·python·spring
wordbaby5 小时前
React Native + RNOH:跨页面数据回传的最佳实践与避坑指南
前端·react native
丷丩5 小时前
MapLibre GL JS第22课:查看本地GeoJSON
前端·javascript·map·mapbox·maplibre gl js
Front思6 小时前
AI前端工程师需要具备能力+
前端·人工智能·ai
ZC跨境爬虫8 小时前
跟着 MDN 学CSS day_29:(掌握文本与字体样式的核心艺术)
前端·css·ui·html·tensorflow
李子琪。9 小时前
网络空间安全深度实战:CSRF 漏洞原理剖析与基于 Token 的纵深防御体系构建(全栈实验报告)
前端·安全·csrf
冰暮流星9 小时前
javascript之history对象介绍
前端·笔记
IT_陈寒9 小时前
Vite热更新失灵?你可能漏了这个配置
前端·人工智能·后端
丷丩9 小时前
MapLibre GL JS第19课:实时更新要素
前端·javascript·gis·map·mapbox·maplibre gl js
Mr.Daozhi9 小时前
RAG 进阶实战:跑通 Demo 后我连续翻了 6 次车,逐一修复才真正可用(含 Gradio Web 版)
前端·数据库·langchain·大模型·gradio·rag·科研工具