iis服务器助手广告广告
返回顶部
首页 > 资讯 > 前端开发 > JavaScript >怎么解决React.useEffect()的无限循环
  • 869
分享到

怎么解决React.useEffect()的无限循环

2024-04-02 19:04:59 869人浏览 八月长安
摘要

这篇文章主要讲解了“怎么解决React.useEffect()的无限循环”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“怎么解决React.useEffect

这篇文章主要讲解了“怎么解决React.useEffect()的无限循环”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“怎么解决React.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中输入任何东西,这是一个无限循环。怎么解决React.useEffect()的无限循环

问题在于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参数中使用对象引用。

  1. let count = 0; 

  2.  

  3. useEffect(() => { 

  4.   // some logic 

  5. }, [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>   ); }

 3 总结

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()时,你还知道有其它方式会引起无限循环陷阱吗?

感谢各位的阅读,以上就是“怎么解决React.useEffect()的无限循环”的内容了,经过本文的学习后,相信大家对怎么解决React.useEffect()的无限循环这一问题有了更深刻的体会,具体使用情况还需要大家实践验证。这里是编程网,小编将为大家推送更多相关知识点的文章,欢迎关注!

--结束END--

本文标题: 怎么解决React.useEffect()的无限循环

本文链接: https://www.lsjlt.com/news/83204.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • 怎么解决React.useEffect()的无限循环
    这篇文章主要讲解了“怎么解决React.useEffect()的无限循环”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“怎么解决React.useEffect...
    99+
    2024-04-02
  • win7开机无限循环重启怎么解决
    如果Windows 7电脑出现无限循环重启的问题,可以尝试以下几种解决方法: 进入安全模式:重启电脑,按F8键进入引导选项菜单,...
    99+
    2024-02-29
    win7
  • C#怎么实现无限循环
    本篇内容主要讲解“C#怎么实现无限循环”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“C#怎么实现无限循环”吧!循环控制语句循环控制语句更改执行的正常序列。当执行离开一个范围时,所有在该范围中创建...
    99+
    2023-06-17
  • Scala无限循环怎么实现
    本篇内容主要讲解“Scala无限循环怎么实现”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Scala无限循环怎么实现”吧!有的时候,我们可能需要多次执行同一块代码。一般情况下,语句是按顺序执行的...
    99+
    2023-06-19
  • 怎么解决React useEffect钩子带来的无限循环问题
    本篇内容主要讲解“怎么解决React useEffect钩子带来的无限循环问题”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“怎么解决React useEffect钩子带来的无...
    99+
    2023-07-02
  • vbs无限循环代码怎么写
    要编写一个VBS脚本的无限循环代码,你可以使用一个无限循环的条件来实现。下面是一个示例:```Do While True' 循环体代...
    99+
    2023-08-12
    vbs
  • win10无限循环登录界面如何解决
    如果您的Windows 10系统出现无限循环登录界面的问题,您可以尝试以下方法来解决:1. 启动修复:在登录界面按住Shift键,同...
    99+
    2023-10-10
    win10
  • win7开机无限循环重启如何解决
    win7开机无限循环重启的原因可能有很多,可能是系统文件损坏、硬件故障、驱动程序冲突等。以下是一些解决方法:1. 进入安全模式:开机...
    99+
    2023-09-01
    win7
  • arduino无线循环问题怎么解决
    要解决Arduino无线循环问题,可以考虑以下几个方面:1. 检查代码逻辑:仔细检查代码中的循环语句,确保没有出现无限循环的情况。如...
    99+
    2023-08-20
    arduino
  • RecyclerView无限循环效果怎么实现
    这篇文章主要介绍“RecyclerView无限循环效果怎么实现”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“RecyclerView无限循环效果怎么实现”文章能帮助大家解决问题。1、修改adpter...
    99+
    2023-07-05
  • Safari5中alert的无限循环BUG
    如下,alert无限循环下去 复制代码 代码如下: <!DOCTYPE htmlt> <html> <head> <meta charset...
    99+
    2022-11-21
    Safari5 alert
  • 如何解决ReactuseEffect钩子带来的无限循环问题
    目录什么导致的无限循环以及如何解决它们如何解决这个问题使用函数作为依赖项使用数组作为依赖项将对象作为依赖项传递传递不正确的依赖项结尾React的useEffect Hook可以让用户...
    99+
    2024-04-02
  • win10自动修复失败无限循环如何解决
    今天小编给大家分享一下win10自动修复失败无限循环如何解决的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。win10自动修复...
    99+
    2023-07-01
  • 怎么使用CSS3实现无限循环的无缝滚动
    小编给大家分享一下怎么使用CSS3实现无限循环的无缝滚动,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!1. 使用CSS3来实现若...
    99+
    2024-04-02
  • CSS3怎么实现无限循环的无缝滚动效果
    这篇文章主要讲解了“CSS3怎么实现无限循环的无缝滚动效果”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“CSS3怎么实现无限循环的无缝滚动效果”吧!1. 使用CSS3来实现若要用CSS3的属...
    99+
    2023-07-05
  • 使用CSS怎么实现背景无缝无限循环
    本篇文章给大家分享的是有关使用CSS怎么实现背景无缝无限循环,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。HTML<div class="dog&qu...
    99+
    2023-06-08
  • vue路由守卫及路由守卫无限循环问题怎么解决
    这篇“vue路由守卫及路由守卫无限循环问题怎么解决”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“vue路由守卫及路由守卫无限...
    99+
    2023-07-04
  • Android中怎么利用ViewPager实现无限循环
    本篇文章为大家展示了Android中怎么利用ViewPager实现无限循环,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。Android ViewPager实现无限循环的实例ViewPager自身并不...
    99+
    2023-05-31
    android viewpager
  • Python中while无限迭代循环怎么实现
    这篇文章主要讲解了“Python中while无限迭代循环怎么实现”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Python中while无限迭代循环怎么实现”吧!前言Python 有 whil...
    99+
    2023-06-30
  • js怎么实现无限循环轮播图效果
    小编给大家分享一下js怎么实现无限循环轮播图效果,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!知识要点1.实现无限循环的原理:以...
    99+
    2024-04-02
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作