你不需要总是在 React 中使用 useState

server/2024/9/22 18:22:48/

在我审查的一个拉取请求中,我注意到在许多拉取请求中看到的一种模式。React 组件具有多个 UI 状态,例如 loadingerrorsuccess

作者使用了多个 useState 钩子来管理这些状态,这导致代码难以阅读且容易出错,例如:

const MyComponent = () => {const [loading, setLoading] = useState(false)const [error, setError] = useState(false)const [success, setSuccess] = useState(false)return (<div>{loading && !error && !success && <p>Loading...</p>}{error && !loading && !success && <p>Error occurred</p>}{success && !loading && !error && <p>Operation completed successfully</p>}</div>)
}

这些状态彼此不同。当 loadingtrue 时,errorsuccess 应该为 false。使用多个 useState 钩子可能会导致意外行为,例如意外 true 同时设置两个状态。

相反,请考虑使用有限状态机(FSM) 模式。FSM 只允许有限数量的状态。在上面的 UI 示例中,单个 useState 可以更稳健地管理当前状态,并且出错的风险更低,如下所示:

import { useState } from 'react'type State = 'loading' | 'error' | 'success'const MyComponent = () => {const [state, setState] = useState<State>('loading')const handleClick = () => {setState('loading')// Simulate an async operationsetTimeout(() => {setState('success')}, 2000)}return (<div>{state === 'loading' && <p>Loading...</p>}{state === 'error' && <p>Error occurred</p>}{state === 'success' && <p>Operation completed successfully</p>}<button onClick={handleClick}>Click me</button></div>)
}

在某些情况下,例如使用 Tanstack 查询来获取数据时,useQuery 无需单独 useState 挂钩 来设置 loadingerrorsuccess 状态:

const MyComponent = () => {const { data, isLoading, error } = useQuery(...)if (isLoading) {return <p>Loading...</p>}if (error) {return <p>Error occurred</p>}return <p>Operation completed successfully {data}</p>
}

让我们考虑另一个名为 locked 的状态,它根据服务器发送的 403 状态代码显示用户是否已解锁该功能。通常情况下,开发人员可能会使用 useStateuseEffect 来管理该状态,这可能会增加不必要的复杂性:

const MyComponent = () => {const [locked, setLocked] = useState(false)const { data, isLoading, error } = useQuery(...)useEffect(() => {if (error && error.status === 403) {setLocked(true)}}, [error])if (locked) {return <p>You are locked out</p>}
}

更好的方法是直接从 error 中推导出锁定状态:

const MyComponent = () => {const { data, isLoading, error } = useQuery(...)if (isLoading) {return <p>Loading...</p>}const locked = error?.status === 403if (locked) {return <p>You are locked out</p>}
}

这种方法可以避免使用 useStateuseEffect 进行额外的状态管理。

在编写 React 组件时,请务必考虑是否有必要使用 useStateuseEffect。通常情况下,它们是不必要的。


http://www.ppmy.cn/server/22588.html

相关文章

每日一题-贪心算法

目录 前言 买入股票的最佳时机(1) 买入股票的最好时机(2) 前言 当你踏上贪心算法的旅程&#xff0c;仿佛置身于一场智慧的盛宴&#xff0c;每一步都是对问题解决方案的审慎选择&#xff0c;每一次决策都是对最优解的向往。贪心算法以其简洁高效的特性&#xff0c;被广泛运用于…

Java基础--单元测试

JUnit是Java中最流行的开源单元测试框架&#xff0c;用于编写和运行可重复的、自动化的单元测试。JUnit极大地简化了测试用例的编写和组织&#xff0c;提供了丰富的断言方法、测试运行控制、测试结果报告等功能&#xff0c;是遵循测试驱动开发&#xff08;TDD&#xff09;和持续…

Vue 组件单元测试深度探索:细致解析与实战范例大全

Vue.js作为一款广受欢迎的前端框架&#xff0c;以其声明式的数据绑定、组件化开发和灵活的生态系统赢得了广大开发者的心。然而&#xff0c;随着项目规模的增长&#xff0c;确保组件的稳定性和可靠性变得愈发关键。单元测试作为软件质量的守护神&#xff0c;为Vue组件的开发过程…

基于Flask的岗位就业可视化系统(一)

前言 本项目综合了基本数据分析的流程&#xff0c;包括数据采集&#xff08;爬虫&#xff09;、数据清洗、数据存储、数据前后端可视化等 推荐阅读顺序为&#xff1a;数据采集——>数据清洗——>数据库存储——>基于Flask的前后端交互&#xff0c;有问题的话可以留言…

Android 在attrs.xml添加属性时出现 Found item Attr/****** more than one time

Android 在attrs.xml添加属性时出现 Found item Attr/****** more than one time 问题描述解决办法方式一方式二 小结 问题描述 在Android应用开发过程中&#xff0c;经常需要自定义控件&#xff0c;并且定义控件的属性&#xff0c;方便灵活的修改控件的显示样式&#xff0c;提…

【数据结构】单链表

单链表 文章目录 单链表定义单链表的优缺点用代码定义单链表初始化单链表不带头结点的单链表带头结点的单链表 单链表的插入按位序插入&#xff08;带头结点&#xff09;指定结点的后插操作指定结点的前插操作 单链表的删除按位序删除&#xff08;带头节点&#xff09;删除指定…

[NeurIPS-23] GOHA: Generalizable One-shot 3D Neural Head Avatar

[pdf | proj | code] 本文提出一种基于单图的可驱动虚拟人像重建框架。基于3DMM给粗重建、驱动结果&#xff0c;基于神经辐射场给细粒度平滑结果。 方法 给定源图片I_s和目标图片I_t&#xff0c;希望生成图片I_o具有源图片ID和目标图片表情位姿。本文提出三个分支&#xff1a;…

C语言-atoi和atof函数的使用

人生应该树立目标&#xff0c;否则你的精力会白白浪费。&#x1f493;&#x1f493;&#x1f493; 目录 •&#x1f319;知识回顾 &#x1f34b;知识点一&#xff1a;atoi函数的使用和实现 • &#x1f330;1.函数介绍 • &#x1f330;2.代码演示 • &#x1f330;3.atoi函数的…