作者:Shadeed
译者:前端小智
来源:dmitripavlutin

点赞再看,微信搜索**【大迁世界】,B站关注【前端小智】**这个没有大厂背景,但有着一股向上积极心态人。本文 GitHub https://github.com/qq449245884/xiaozhi 上已经收录,文章的已分类,也整理了很多我的文档,和教程资料。

最近开源了一个 Vue 组件,还不够完善,欢迎大家来一起完善它,也希望大家能给个 star 支持一下,谢谢各位了。

github 地址:https://github.com/qq449245884/vue-okr-tree

useEffect() 主要用来管理副作用,比如通过网络抓取、直接操作 DOM、启动和结束计时器。

虽然useEffect()useState(管理状态的方法)是最常用的钩子之一,但需要一些时间来熟悉和正确使用。

使用useEffect()时,你可能会遇到一个陷阱,那就是组件渲染的无限循环。在这篇文章中,会讲一下产生无限循环的常见场景以及如何避免它们。

1. 无限循环和副作用更新状态

假设我们有一个功能组件,该组件里面有一个 input 元素,组件是功能是计算 input 更改的次数。

我们给这个组件取名为 CountInputChanges,大概的内容如下:

function CountInputChanges() {const [value, setValue] = useState('');const [count, setCount] = useState(-1);useEffect(() => setCount(count + 1));const onChange = ({ target }) => setValue(target.value);return (<div><input type="text" value={value} onChange={onChange} /><div>Number of changes: {count}</div></div>)
}

<input type =“ text” value = {value} onChange = {onChange} />是受控组件。value变量保存着 input 输入的值,当用户输入输入时,onChange事件处理程序更新 value 状态。

这里使用useEffect()更新count变量。每次由于用户输入而导致组件重新渲染时,useEffect(() => setCount(count + 1))就会更新计数器。

因为useEffect(() => setCount(count + 1))是在没有依赖参数的情况下使用的,所以()=> setCount(count + 1)会在每次渲染组件后执行回调。

你觉得这样写会有问题吗?打开演示自己试试看:https://codesandbox.io/s/infinite-loop-9rb8c?file=/src/App.js

运行了会发现count状态变量不受控制地增加,即使没有在input中输入任何东西,这是一个无限循环。

问题在于useEffect()的使用方式:

useEffect(() => setCount(count + 1));

它生成一个无限循环的组件重新渲染。

在初始渲染之后,useEffect()执行更新状态的副作用回调函数。状态更新触发重新渲染。重新渲染之后,useEffect()执行副作用回调并再次更新状态,这将再次触发重新渲染。

1.1通过依赖来解决

无限循环可以通过正确管理useEffect(callback, dependencies)依赖项参数来修复。

因为我们希望count在值更改时增加,所以可以简单地将value作为副作用的依赖项。

import { useEffect, useState } from 'react';function CountInputChanges() {const [value, setValue] = useState('');const [count, setCount] = useState(-1);useEffect(() => setCount(count + 1), [value]);const onChange = ({ target }) => setValue(target.value);return (<div><input type="text" value={value} onChange={onChange} /><div>Number of changes: {count}</div></div>);
}

添加[value]作为useEffect的依赖,这样只有当[value]发生变化时,计数状态变量才会更新。这样做可以解决无限循环。

1.2 使用 ref

除了依赖,我们还可以通过 useRef() 来解决这个问题。

其思想是更新 Ref 不会触发组件的重新渲染。

import { useEffect, useState, useRef } from "react";function CountInputChanges() {const [value, setValue] = useState("");const countRef = useRef(0);useEffect(() => countRef.current++);const onChange = ({ target }) => setValue(target.value);return (<div><input type="text" value={value} onChange={onChange} /><div>Number of changes: {countRef.current}</div></div>);
}

useEffect(() => countRef.current++) 每次由于value的变化而重新渲染后,countRef.current++就会返回。引用更改本身不会触发组件重新渲染。

2. 无限循环和新对象引用

即使正确设置了useEffect()依赖关系,使用对象作为依赖关系时也要小心。

例如,下面的组件CountSecrets监听用户在input中输入的单词,一旦用户输入特殊单词'secret',统计 ‘secret’ 的次数就会加 1。

import { useEffect, useState } from "react";function CountSecrets() {const [secret, setSecret] = useState({ value: "", countSecrets: 0 });useEffect(() => {if (secret.value === 'secret') {setSecret(s => ({...s, countSecrets: s.countSecrets + 1}));    }}, [secret]);const onChange = ({ target }) => {setSecret(s => ({ ...s, value: target.value }));};return (<div><input type="text" value={secret.value} onChange={onChange} /><div>Number of secrets: {secret.countSecrets}</div></div>);
}

打开演示(https://codesandbox.io/s/infinite-loop-obj-dependency-7t26v?file=/src/App.js)自己试试,当前输入 secretsecret.countSecrets的值就开始不受控制地增长。

这是一个无限循环问题。

为什么会这样?

secret对象被用作useEffect(..., [secret])。在副作用回调函数中,只要输入值等于secret,就会调用更新函数

setSecret(s => ({...s, countSecrets: s.countSecrets + 1}));

这会增加countSecrets的值,但也会创建一个新对象。

secret现在是一个新对象,依赖关系也发生了变化。所以useEffect(..., [secret])再次调用更新状态和再次创建新的secret对象的副作用,以此类推。

JavaScript 中的两个对象只有在引用完全相同的对象时才相等。

2.1 避免将对象作为依赖项

解决由循环创建新对象而产生的无限循环问题的最好方法是避免在useEffect()dependencies参数中使用对象引用。

let count = 0;useEffect(() => {// some logic
}, [count]); // Good!
let myObject = {prop: 'Value'
};useEffect(() => {// some logic
}, [myObject]); // Not good!
useEffect(() => {// some logic
}, [myObject.prop]); // Good!

修复<CountSecrets>组件的无限循环问题,可以将useEffect(..., [secret])) 变为 useEffect(..., [secret.value])

仅在secret.value更改时调用副作用回调就足够了,下面是修复后的代码:

import { useEffect, useState } from "react";function CountSecrets() {const [secret, setSecret] = useState({ value: "", countSecrets: 0 });useEffect(() => {if (secret.value === 'secret') {setSecret(s => ({...s, countSecrets: s.countSecrets + 1}));}}, [secret.value]);const onChange = ({ target }) => {setSecret(s => ({ ...s, value: target.value }));};return (<div><input type="text" value={secret.value} onChange={onChange} /><div>Number of secrets: {secret.countSecrets}</div></div>);
}

3 总结

useEffect(callback, deps)是在组件渲染后执行callback(副作用)的 Hook。如果不注意副作用的作用,可能会触发组件渲染的无限循环。

生成无限循环的常见情况是在副作用中更新状态,没有指定任何依赖参数

useEffect(() => {// Infinite loop!setState(count + 1);
});

避免无限循环的一种有效方法是正确设置依赖项:

useEffect(() => {// No infinite loopsetState(count + 1);
}, [whenToUpdateValue]);

另外,也可以使用 Ref,更新 Ref 不会触发重新渲染:

useEffect(() => {// No infinite loopcountRef.current++;
});

无限循环的另一种常见方法是使用对象作为useEffect()的依赖项,并在副作用中更新该对象(有效地创建一个新对象)

useEffect(() => {// Infinite loop!setObject({...object,prop: 'newValue'})
}, [object]);

避免使用对象作为依赖项,只使用特定的属性(最终结果应该是一个原始值):

useEffect(() => {// No infinite loopsetObject({...object,prop: 'newValue'})
}, [object.whenToUpdateProp]);

当使用useEffect()时,你还知道有其它方式会引起无限循环陷阱吗?

完,我是小智,我们下期见


代码部署后可能存在的BUG没法实时知道,事后为了解决这些BUG,花了大量的时间进行log 调试,这边顺便给大家推荐一个好用的BUG监控工具 Fundebug。

原文:https://dmitripavlutin.com/react-useeffect-infinite-loop/

交流

文章每周持续更新,可以微信搜索「 大迁世界 」第一时间阅读和催更(比博客早一到两篇哟),本文 GitHub https://github.com/qq449245884/xiaozhi 已经收录,整理了很多我的文档,欢迎Star和完善,大家面试可以参照考点复习,另外关注公众号,后台回复福利,即可看到福利,你懂的。

如何解决 React.useEffect() 的无限循环相关推荐

  1. vue本地没事放到服务器上无限循环,解决vue中的无限循环问题

    项目中遇到了这样一个问题:每一种产品有对应的服务费,每一个商家有多种商品要单独计算每一家的服务费,最后汇总总的服务费用.我直接写了一个方法来计算出每个商家和总的服务费用并return出来.如果不看控制 ...

  2. 3 种导致 React 无限循环的方式

    原文:3 ways to cause an infinite loop in React https://alexsidorenko.com/blog/react-infinite-loop/ Rea ...

  3. php easyui tree 结构,EasyUI Tree树组件无限循环的解决方法

    在学习jquery easyui的tree组件的时候,在url为链接地址的时,发现如果最后一个节点的state为closed时,未节点显示为文件夹,单击会重新加载动态(Url:链接地址)形成无限循环. ...

  4. linux系统一直循环登录界面,Ubuntu 14.04解决登录界面无限循环的方法

    在Ubuntu下配置Android的环境时,想像在Windows中那样在终端中直接启动adb,以为Linux和Windows一样,将adb的路径添加到环境变量中,于是将adb的路径也export到/e ...

  5. win10无限重启_win10系统安装无限循环如何解决_win10教程

    如果您因为使用在线重装工具造成了win10系统安装无限循环问题,那么不用怕,跟着小编一起解决它吧.您目前需要再找一台能正常使用的电脑,一个 4GB 以上的空白 U 盘,制作一个启动盘即可完成系统重装. ...

  6. 返回JSON出现Infinite recursion无限循环错误的解决

    在数据库返回树形结构数据之后,想转换成JSON返回页面时出错: org.springframework.http.converter.HttpMessageNotWritableException: ...

  7. web项目中实现登陆成功后才能访问某些页面,否则自动跳转到登陆界面,以及对无限循环重定向的解决

    1.不使用过滤器实现,在需要拦截的页面通过session判断是否登陆,若没登陆则跳转到login.jsp页面: <% Admin admin = (Admin)session.getAttrib ...

  8. 如何修复计算机的无限功能,Win10专业版自动修复失败无限循环的解决方法

    Win10专业版自动修复失败无限循环如何解决?Win10是个完整的系统,不仅自带各种应用,如计算器.画图等,还有系统重置.自动修复等功能,在系统瘫痪时使用系统自带的工具来修复.还原.Win10的自动修 ...

  9. win10更新后输入密码后无限循环解决方法

    win10更新后输入密码后无限循环解决方法 1.解决方法 2.原因解析 3.其他说明 win10更新后输入密码后无限循环解决方法 1.解决方法 1.网上查询到的解决方法都是重启进入安全模式,然后选择疑 ...

最新文章

  1. php内打开网址,网站内部跳转外部网站go.php
  2. python语言入门w-Python 基础教程
  3. php url 调度
  4. CentOS 安装MySQL5.7 源码方式安装
  5. 下载网络图片显示在Android手机上
  6. php升维,svm算法详解
  7. HDU2078 复习时间【水题】
  8. sever串口wifi拓展板_什么是串口WIFI模块
  9. NXP Kinetis的 单片机LPUART 模块调试记录
  10. 查看GitHub仓库大小的几种方法
  11. redission限流RedisException问题排查
  12. Huffman 编码的实现(C语言)
  13. AI绘图实战(一):制作购物车图标icon | Stable Diffusion成为设计师生产力工具
  14. 解密秒杀系统架构:不是所有的秒杀都是秒杀
  15. 收支两条线资金管理模式的应用
  16. 移远 BC35-G 模块固件升级
  17. confluence 使用教程
  18. 游戏美术行业的薪资待遇,人与人的差距真的在于努力,来看看数据吧
  19. 土地利用总体规划数据库问题
  20. 关于GOG Galaxy 2.0提示“无法下载所需文件”问题的解决方案:

热门文章

  1. warframe怒焚者结合目标_warframe大黄脸扫描器简介 星际战甲结合扫描器效果一览...
  2. 林内大智系列有服务器,林内热水器有哪些型号 林内热水器哪个型号好【详解】...
  3. 62、63 不同路径
  4. 使用selenium,xpath,线程池爬取斗鱼主播信息
  5. Docker搭建私有镜像仓库
  6. 无代码平台与SaaS(含平台评估推荐)
  7. 从十大经典故事中学员工管理和激励
  8. 制作人物双重曝光海报
  9. Vi设计的可能的行经
  10. 绝地求生改文件选服务器,绝地求生怎么改文件位置 | 手游网游页游攻略大全