文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

如何解决 React.useEffect() 的无限循环

2024-12-03 07:16

关注

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

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

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

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

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

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

  1. function CountInputChanges() { 
  2.   const [value, setValue] = useState(''); 
  3.   const [count, setCount] = useState(-1); 
  4.  
  5.  useEffect(() => setCount(count + 1)); 
  6.   const onChange = ({ target }) => setValue(target.value); 
  7.  
  8.   return ( 
  9.     
     
  10.  "text" value={value} onChange={onChange} /> 
  11.  
    Number of changes: {count}
     
  12.  
 
  •   ) 
  •   是受控组件。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()的使用方式:

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

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

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


    1.1通过依赖来解决

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

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

    1. import { useEffect, useState } from 'react'
    2.  
    3. function CountInputChanges() { 
    4.   const [value, setValue] = useState(''); 
    5.   const [count, setCount] = useState(-1); 
    6.  
    7.  useEffect(() => setCount(count + 1), [value]); 
    8.   const onChange = ({ target }) => setValue(target.value); 
    9.  
    10.   return ( 
    11.     
       
    12.  "text" value={value} onChange={onChange} /> 
    13.  
      Number of changes: {count}
       
    14.  
     
  •   ); 
  •  添加[value]作为useEffect的依赖,这样只有当[value]发生变化时,计数状态变量才会更新。这样做可以解决无限循环。


    1.2 使用 ref

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

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

    1. import { useEffect, useState, useRef } from "react"
    2.  
    3. function CountInputChanges() { 
    4.   const [value, setValue] = useState(""); 
    5.   const countRef = useRef(0); 
    6.  
    7.  useEffect(() => countRef.current++); 
    8.   const onChange = ({ target }) => setValue(target.value); 
    9.  
    10.   return ( 
    11.     
       
    12.  "text" value={value} onChange={onChange} /> 
    13.  
      Number of changes: {countRef.current}
       
    14.   
    15.   ); 

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

     

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

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

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

    1. import { useEffect, useState } from "react"
    2.  
    3. function CountSecrets() { 
    4.   const [secret, setSecret] = useState({ value: "", countSecrets: 0 }); 
    5.  
    6.   useEffect(() => { 
    7.     if (secret.value === 'secret') { 
    8.  setSecret(s => ({...s, countSecrets: s.countSecrets + 1}));    } 
    9.  }, [secret]); 
    10.   const onChange = ({ target }) => { 
    11.     setSecret(s => ({ ...s, value: target.value })); 
    12.   }; 
    13.  
    14.   return ( 
    15.     
       
    16.  "text" value={secret.value} onChange={onChange} /> 
    17.  
      Number of secrets: {secret.countSecrets}
       
    18.   
    19.   ); 

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

    这是一个无限循环问题。

    为什么会这样?

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

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

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

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

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

    2.1 避免将对象作为依赖项

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

    1. let count = 0; 
    2.  
    3. useEffect(() => { 
    4.   // some logic 
    5. }, [count]); // Good! 

    1. let myObject = { 
    2.   prop: 'Value' 
    3. }; 
    4.  
    5. useEffect(() => { 
    6.   // some logic 
    7. }, [myObject]); // Not good! 
    8. useEffect(() => { 
    9.   // some logic 
    10. }, [myObject.prop]); // Good! 

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

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

    1. import { useEffect, useState } from "react"
    2.  
    3. function CountSecrets() { 
    4.   const [secret, setSecret] = useState({ value: "", countSecrets: 0 }); 
    5.  
    6.   useEffect(() => { 
    7.     if (secret.value === 'secret') { 
    8.       setSecret(s => ({...s, countSecrets: s.countSecrets + 1})); 
    9.     } 
    10.  }, [secret.value]); 
    11.   const onChange = ({ target }) => { 
    12.     setSecret(s => ({ ...s, value: target.value })); 
    13.   }; 
    14.  
    15.   return ( 
    16.     
       
    17.  "text" value={secret.value} onChange={onChange} /> 
    18.  
      Number of secrets: {secret.countSecrets}
       
    19.   
    20.   ); 

     3 总结

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

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

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

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

    1. useEffect(() => { 
    2.   // No infinite loop 
    3.   setState(count + 1); 
    4. }, [whenToUpdateValue]); 

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

    1. useEffect(() => { 
    2.   // No infinite loop 
    3.   countRef.current++; 
    4. }); 

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

    1. useEffect(() => { 
    2.   // Infinite loop! 
    3.   setObject({ 
    4.     ...object, 
    5.     prop: 'newValue' 
    6.   }) 
    7. }, [object]); 

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

    1. useEffect(() => { 
    2.   // No infinite loop 
    3.   setObject({ 
    4.     ...object, 
    5.     prop: 'newValue' 
    6.   }) 
    7. }, [object.whenToUpdateProp]); 

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

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

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

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

     

    来源: 大迁世界内容投诉

    免责声明:

    ① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

    ② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

    软考中级精品资料免费领

    • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

      难度     813人已做
      查看
    • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

      难度     354人已做
      查看
    • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

      难度     318人已做
      查看
    • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

      难度     435人已做
      查看
    • 2024年上半年系统架构设计师考试综合知识真题

      难度     224人已做
      查看

    相关文章

    发现更多好内容

    猜你喜欢

    AI推送时光机
    位置:首页-资讯-后端开发
    咦!没有更多了?去看看其它编程学习网 内容吧
    首页课程
    资料下载
    问答资讯