首页 > 其他分享 >如何解决 React.useEffect() 的无限循环

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

时间:2023-07-13 18:46:17浏览次数:50  
标签:count const secret value React 无限 组件 useEffect

[如何解决 React.useEffect() 的无限循环-react.useeffect](https://www.51cto.com/article/654928.html)

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

 

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)自己试试,当前输入 secret,secret.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! 

 

修复组件的无限循环问题,可以将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> 
  ); 
} 

     

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

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

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

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

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

     

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

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

     

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

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

     

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

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

     

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

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

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

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

    标签:count,const,secret,value,React,无限,组件,useEffect
    From: https://www.cnblogs.com/ministep/p/17551784.html

    相关文章

    • react-d3-tree自定义节点使用案例
      react-d3-tree主要API及其中文解释:Tree组件的props:这些API提供了丰富的配置选项,可以用来定制树的外观和行为。例如,可以使用nodeSize属性调整节点的大小,使用pathFunc属性绘制自定义的连线,使用onClick属性处理节点的点击事件等等。data:树的数据对象。zoomable:指......
    • WPF开发中ReactiveUI.Fody的使用
      前面的开发一般我会使用PropertyChanged.Fody,但ReactiveUI.Fody也能实现类似的功能。安装Nuget包Install-PackageReactiveUI.FodyFodyWeavers.xml文件:<Weaversxmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:noNamespaceSchemaLocation="FodyWeavers.xsd"......
    • react 内容
      1、初始安装react+ts npx create-react-appmy-app  npxcreate-react-appreact-ts --templatetypescript 安装为ts2、安装好react后面加上ts 2.1tsc--init 生产tsconfig.js{  "compilerOptions": {    "target": "es5" /**指定ECMAScrip......
    • React中编写操作树形数据的自定义Hook
      什么是Hookhook即为钩子,是一种特殊的函数,它可以让你在函数式组件中使用一些react特性,目前在react中常用的hook有以下几类useState:用于在函数组件中定义和使用状态(state)。useEffect:用于在函数组件中处理副作用,也可以模拟react生命周期useContext:用于在函......
    • shallowReactive 与 shallowRef
      shallowReactive:只处理对象最外层属性的响应式(浅响应式)。shallowRef:只处理基本数据类型的响应式,不进行对象的响应式处理。什么时候使用?如果有一个对象数据,结构比较深,但变化时只是外层属性变化===>shallowReactive。如果有一个对象数据,后续功能不会修改该对......
    • 创新体验的未来:探索VR与数字孪生的无限可能性
      VR即虚拟现实(VirtualReality)和数字孪生(DigitalTwin)是当今科技领域的两大热门概念,VR以其沉浸式的体验和逼真的虚拟环境,让用户身临其境,开启了全新的交互方式和感官体验;而数字孪生则以其复制现实世界的精确模型,为现实世界提供了虚拟的镜像,实现了真实与虚拟的融合。 VR技术的出现......
    • 前端框架及项目面试-聚焦Vue、React、Webpack
      第1章课程导学介绍课程制作的背景和课程主要内容。第2章课程介绍先出几道面试真题,引导思考。带着问题来继续学习,效果更好。第3章Vue使用Vue是前端面试必考内容,首先要保证自己要会使用Vue。本章讲解Vue基本使用、组件使用、高级特性和VuexVue-router,这些部分的知识点和......
    • React18+Next.js13+TS,B端+C端完整业务+技术双闭环(20章)
      最新React技术栈,实战复杂低代码项目-仿问卷星第1章开期准备试看3节|20分钟介绍课程内容,学习建议和注意事项。演示课程项目,让学员有一个整体的认识。第2章【入门】什么是ReactReact引领了现代前端开发的变革8节|50分钟介绍React的历史、背景和每次版本更新。介绍R......
    • 客服react
      //import{useEffect,useState}from"react";//exportdefaultfunctionceshi(){//const[value,setValue]=useState(2);//const[data,setData]=useState('');//constws=newWebSocket('ws://124.222.224.186:8800�......
    • 《ReAct: SYNERGIZING REASONING AND ACTING IN LANGUAGE MODELS》论文学习
      一、论文主要思想本文首先认为,到目前为止,LLM在语言理解方面令人印象深刻,它们已被用来生成CoT(思想链)来解决一些问题,它们也被用于执行和计划生成。尽管这两者是分开研究的,但本文旨在以交错的方式将推理和行动结合起来,以提高LLM的表现。这个想法背后的原因是,如果你考虑一下作为......