使用React Hooks 时要避免的5个错误!,前端开发项目

来源:dmitripavlutin

很有可能你已经读过很多关于如何使用React Hook 的文章。但有时候,知道何时不使用与知道如何使用同样重要。

在这篇文章中,主要介绍一下 React hooks 错误使用方式,以及如何解决它们。

  • 不要更改 Hook 调用顺序

  • 不要使用过时状态

  • 不要创建过时的闭包

  • 不要将状态用于基础结构数据

  • 不要忘记清理副作用

1.不要更改 Hook 调用顺序

在写这篇文章的前几天,我编写了一个通过id获取游戏信息的组件,下面是一个简单的版本 FetchGame

function FetchGame({ id }) {

if (!id) {

return ‘Please select a game to fetch’;

}

const [game, setGame] = useState({

name: ‘’,

description: ‘’

});

useEffect(() => {

const fetchGame = async () => {

const response = await fetch(/api/game/${id});

const fetchedGame = await response.json();

setGame(fetchedGame);

};

fetchGame();

}, [id]);

return (

Name: {game.name}
Description: {game.description}

);

组件FetchGame 接收 id(即要获取的游戏的ID)。 useEffect()await fetch(/game/${id})提取游戏信息并将其保存到状态变量game中。

打开演示(https://codesandbox.io/s/hooks-order-warning-rdxpg?file=/pages/index.js) 。组件正确地执行获取操作,并使用获取的数据更新状态。但是看看tab Eslint警告: 有 Hook 执行顺序不正确的问题。

问题发生在这一判断:

function FetchGame({ id }) {

if (!id) { return ‘Please select a game to fetch’; }

// …

}

id为空时,组件渲染'Please select a game to fetch'并退出,不调用任何 Hook。

但是,如果 id不为空(例如等于’1’),则会调用useState()useEffect()

有条件地执行 Hook 可能会导致难以调试的意外错误。React Hook的内部工作方式要求组件在渲染之间总是以相同的顺序调用 Hook。

这正是钩子的第一条规则:不要在循环、条件或嵌套函数内调用 Hook。

解决方法就是将条件判断放到 Hook 后面:

function FetchGame({ id }) {

const [game, setGame] = useState({

name: ‘’,

description: ‘’

});

useEffect(() => {

const fetchGame = async () => {

const response = await fetch(/api/game/${id});

const fetchedGame = await response.json();

setGame(fetchedGame);

};

if (id) { fetchGame(); } }, [id]);

if (!id) { return ‘Please select a game to fetch’; }

return (

Name: {game.name}
Description: {game.description}

);

}

现在,无论id是否为空,useState()useEffect() 总是以相同的顺序被调用,这就是 Hook 应该始终被调用的方式。

2.不要使用过时状态

下面的组件MyIncreaser在单击按钮时增加状态变量count:

function MyIncreaser() {

const [count, setCount] = useState(0);

const increase = useCallback(() => {

setCount(count + 1);

}, [count]);

const handleClick = () {

increase(); increase(); increase(); };

return (

<>

Increase

Counter: {count}

</>

);

}

这里有趣一点的是,handleClick调用了3次状态更新。

现在,在打开演示之前,问一个问题: 如果单击一次按钮,计数器是否增加3

打开演示(https://codesandbox.io/s/stale-variable-jo32q?file=/src/index.js),点击按钮一次,看看结果。

不好意思,即使在handleClick()中3次调用了increase(),计数也只增加了1

问题在于setCount(count + 1)状态更新器。当按钮被点击时,React调用setCount(count + 1)3次

const handleClick = () {

increase();

increase();

increase();

};

// 等价:

const handleClick = () {

setCount(count + 1);

// count variable is now stale

setCount(count + 1);

setCount(count + 1);

};

setCount(count + 1)的第一次调用正确地将计数器更新为count + 1 = 0 + 1 = 1。但是,接下来的两次setCount(count + 1)调用也将计数设置为1,因为它们使用了过时的stale状态。

通过使用函数方式更新状态来解决过时的状态。我们用setCount(count => count + 1)代替setCount(count + 1)

function MyIncreaser() {

const [count, setCount] = useState(0);

const increase = useCallback(() => {

setCount(count => count + 1); }, []);

const handleClick = () {

increase();

increase();

increase();

};

return (

<>

Increase

Counter: {count}

</>

);

}

这里有一个好规则可以避免遇到过时的变量:

如果你使用当前状态来计算下一个状态,总是使用函数方式来更新状态:setValue(prevValue => prevValue + someResult)

3.不要创建过时的闭包

React Hook 很大程序上依赖于闭包的概念。依赖闭包是它们如此富有表现力的原因。

JavaScript 中的闭包是从其词法作用域捕获变量的函数。不管闭包在哪里执行,它总是可以从定义它的地方访问变量。

当使用 Hook 接受回调作为参数时(如useEffect(callback, deps)useCallback(callback, deps)),你可能会创建一个过时的闭包,一个捕获了过时的状态或变量的闭包。

我们来看看一个使用useEffect(callback, deps) 而忘记正确设置依赖关系时创建的过时闭包的例子。

在组件<WatchCount>中,useEffect()每2秒打印一次count的值

const [count, setCount] = useState(0);

useEffect(function() {

setInterval(function log() {

console.log(Count is: ${count});

}, 2000);

}, []);

const handleClick = () => setCount(count => count + 1);

return (

<> Increase

Counter: {count}
</>

);

}

打开演示(https://codesandbox.io/s/stale-variable-jo32q?file=/src/index.js),点击按钮。在控制台查看,每2秒打印的都 是 Count is: 0,,不管count状态变量的实际值是多少。

为啥这样子?

第一次渲染时, log 函数捕获到的 count 的值为 0

之后,当按钮被单击并且count增加时,setInterval取到的 count 值仍然是从初始渲染中捕获count为0的值。log 函数是一个过时的闭包,因为它捕获了一个过时的状态变量count

解决方案是让useEffect()知道闭包log依赖于count,并正确重置计时器

function WatchCount() {

const [count, setCount] = useState(0);

useEffect(function() {

const id = setInterval(function log() {

console.log(Count is: ${count});

}, 2000);

return () => clearInterval(id); }, [count]);

const handleClick = () => setCount(count => count + 1);

return (

<>

Increase

Counter: {count}

</>

);

}

正确设置依赖关系后,一旦count发生变化,useEffect()就会更新setInterval()的闭包。

为了防止闭包捕获旧值:确保提供给 Hook 的回调函数中使用依赖项。

4.不要将状态用于基础结构数据

有一次,我需要在状态更新上调用副作用,在第一个渲染不用调用副作用。useEffect(callback, deps)总是在挂载组件后调用回调函数:所以我想避免这种情况。

我找到了以下的解决方案

function MyComponent() {

const [isFirst, setIsFirst] = useState(true);

const [count, setCount] = useState(0);

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数前端工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Web前端开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上前端开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:前端)

最后

你要问前端开发难不难,我就得说计算机领域里常说的一句话,这句话就是『难的不会,会的不难』,对于不熟悉某领域技术的人来说,因为不了解所以产生神秘感,神秘感就会让人感觉很难,也就是『难的不会』;当学会这项技术之后,知道什么什么技术能做到什么做不到,只是做起来花多少时间的问题而已,没啥难的,所以就是『会的不难』。

我特地针对初学者整理一套前端学习资料,免费分享给大家,戳这里即可免费领取

前端路线图

%以上前端开发知识点,真正体系化!**

[外链图片转存中…(img-I58ca1dA-1712296953791)]

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:前端)

最后

你要问前端开发难不难,我就得说计算机领域里常说的一句话,这句话就是『难的不会,会的不难』,对于不熟悉某领域技术的人来说,因为不了解所以产生神秘感,神秘感就会让人感觉很难,也就是『难的不会』;当学会这项技术之后,知道什么什么技术能做到什么做不到,只是做起来花多少时间的问题而已,没啥难的,所以就是『会的不难』。

我特地针对初学者整理一套前端学习资料,免费分享给大家,戳这里即可免费领取

[外链图片转存中…(img-vnoUfB71-1712296953791)]

vue.js的36个技巧

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值