React18的useEffect会执行两次

这篇具有很好参考价值的文章主要介绍了React18的useEffect会执行两次。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

一、执行两次的useEffect。

前段时间在本地启了一个 React Demo 项目,在编码的过程中遇到一个很奇怪的“Bug”。
其中简化版的代码如下所示。

// 入口文件
import { StrictMode } from 'react';
import * as ReactDOMClient from 'react-dom/client';
import App from './App';
const root = ReactDOMClient.createRoot(document.getElementById('root'));
root.render(
  <StrictMode>
    <App />
  </StrictMode>
);
// 组件代码
import React, { useEffect } from 'react';

const App = () => {
  useEffect(() => {
    console.log('组件挂载完成!');
  }, []);
  return <>Hello world!</>;
};

我是万万没想到,就这样几行简单的代码竟然会触发一个“Bug”。
此“Bug”的表现为:在 Chrome 控制台里发现 “Hello world!” 被打印了 “两次”。
刷新之后依然如此,当时就给我整懵了,第一感觉就是,这怎么可能?
很是纠结一番之后依然没想明白,于是试着去网上搜了一下,发现竟然有人同样遇到过这个问题。

通过网上指引,同时去官网查了一下,终于得出答案。

这不是 Bug,这是 React18 新加的特性。

二、React18 useEffect 新特性

1.这是 React18 才新增的特性。
2.仅在开发模式("development")下,且使用了严格模式("Strict Mode")下会触发。
  生产环境("production")模式下和原来一样,仅执行一次。
3.之所以执行两次,是为了模拟立即卸载组件和重新挂载组件。
  为了帮助开发者提前发现重复挂载造成的 Bug 的代码。 
  同时,也是为了以后 React的新功能做铺垫。 
  未来会给 React 增加一个特性,允许 React 在保留状态的同时,能够做到仅仅对UI部分的添加和删除。
  让开发者能够提前习惯和适应,做到组件的卸载和重新挂载之后, 重复执行 useEffect的时候不会影响应用正常运行。

如何应对

看过文档以及了解他们这么做的本意之后,我也能够理解他们会这样做了。
只是,对于这种半强迫式操作多少有些不喜欢,感觉是在代码中”被强迫打一针疫苗?”。

当然,人家就是这么干了,作为 React 的普通使用者,能做的就是 适应它 ,并按照它的规范来做。

1.首先先了解一下 React 中 useEffect 执行的时机

Every time your component renders, React will update the screen and then run the
code inside useEffect.

每次组件渲染时,React 都会更新页面 UI,然后运行 useEffect 中的代码。

Effects run at the end of the rendering process after the screen updates

Effect 在屏幕更新之后的 rendering 进程结束的时候执行。

从上面可以得出结论,React 中的 useEffect 执行时机是在组件渲染之后(类似于 window(component).onload ?)。
因此,对于某些“副作用”的渲染,比如异步接口请求,事件绑定等操作我们通常都放在 useEffect 中执行。

当然,useEffect 除了在组件渲染的时候执行外,在组件卸载的时候也有相关执行操作。
在组件卸载的时候会执行 useEffect 方法的return语句。

useEffect(() => {
  window.a = 100;
  return (window.a = 0);
}, []);

如上代码段,当组件渲染的时候会执行window.a = 100,当组件卸载的时候会执行window.a = 0。

知道了 useEffect 的执行时机,也就能明白为什么 React18 中 useEffect 会执行两次了。

因为, React18 在开发环境中除了必要的挂载之外,还 "额外"模拟执行了一次组件的卸载和挂载。

既然知道了原因,那么,接下来就是想办法解决了。

2.怎么样才能让 Effect 执行一次?。

对于这个问题,官方文档上面有一句原话:The right question isn’t “how to run an Effect once,” but “how to fix my Effect so that it works after remounting”.翻译一下,就是说:正确的问题不是“怎么样让 Effect 执行一次”,而是“怎样修复我的 Effect,让它在(重复)挂载之后正常工作”
也可以理解,毕竟在 React 的未来版本中做离屏渲染的时候 useEffect 肯定会多次执行的。
而且,即使是当前版本,在做页面的前进后退也会面临触发多次 useEffect。

所以,解决办法其实就是解决 重复挂载卸载之后 应用正常工作了。
###3.具体的解决方法
我们知道 useEffect 支持返回一个函数,在组件卸载的时候就会执行该函数。
因此,通常正确解法就是 实现清理函数,并将其在 useEffect 中返回。
当然,不同的 Effect 需要有不同的清理方式。

在常用 Effect 分类下,大致有如下几类清理。

1)清理事件监听
useEffect(() => {
  function handleScroll(e) {
    console.log(e.clientX, e.clientY);
  }
  window.addEventListener('scroll', handleScroll);
  return () => window.removeEventListener('scroll', handleScroll);
}, []);

对于事件监听类函数,在返回函数内部“取消掉事件监听”即可。

2-1)重置页面数据,清理属性状态
useEffect(() => {
  const node = ref.current;
  node.style.opacity = 1; // Trigger the animation
  return () => {
    node.style.opacity = 0; // Reset to the initial value
  };
}, []);

对于一些页面属性的变更,在返回函数内部将其变更的属性进行还原。

2-2)重置页面数据,还原元素状态
import { useEffect, useRef } from 'react';

function VideoPlayer({ src, isPlaying }) {
  const ref = useRef(null);

  useEffect(() => {
    if (isPlaying) {
      ref.current.play();
    } else {
      ref.current.pause();
    }
  });

  return <video ref={ref} src={src} loop playsInline />;
}

涉及到元素状态的,比如播放器之类,需要对(元素)播放器的状态进行重置。

2-3)重置页面数据,弹窗类。
useEffect(() => {
  const dialog = dialogRef.current;
  dialog.showModal();
  return () => dialog.close();
}, []);

如果是默认弹窗类,这种也算是元素状态,同样需要对其(弹出)状态进行重置。

3-1)异步请求页面数据处理,处理异步数据渲染
useEffect(() => {
  let ignore = false;
  async function startFetching() {
    const json = await fetchTodos(userId);
    // 这里执行是异步的,所以第一次执行到此处的时候组件已经被卸载了
    // 此时的 ignore 已经被 return 里面的方法置为 true 了
    // 所以这里第一次执行的时候不执行 setTodos(json)
    // setTodos 其实是在第二次执行的时候才触发
    if (!ignore) {
      setTodos(json);
    }
  }
  startFetching();

  return () => {
    ignore = true;
  };
}, [userId]);

如上代码,对于异步请求数据并渲染这一类。
我们可以设置一个 标识位,做到对 请求返回的数据 仅做一次处理与渲染setTodos(json)。
codesandbox 测试代码段

3-2)异步请求页面数据处理,处理接口请求

上面的方法虽然仅会渲染一次,但是请求依然发起了多次。
如果不希望请求多次,也可以使用请求接口数据的缓存方案,对返回数据进行缓存。

const cache = useRef(null);
useEffect(() => {
  let ignore = false;
  async function startFetching() {
    if (!cache.current) {
      cache.current = await fetchTodos(userId);
    }
    if (!ignore) {
      setTodos(cache.current);
    }
  }
  startFetching();

  return () => {
    ignore = true;
  };
}, [userId]);

对于异步请求,除了可以处理渲染频率,还可以对接口的请求本身做缓存。
在前面3-1的基础上,缓存接口返回的数据,下次请求的时候如果已经有缓存数据了就直接用,无须再次发起请求。

4)无须清理类

并不是所有的 useEffect 函数都需要清理,对于一些没有副作用的函数,我们完全可以不做处理

useEffect(() => {
  const map = mapRef.current;
  map.setZoomLevel(zoomLevel);
}, [zoomLevel]);

如上代码所示,setZoomLevel 方法仅仅是设置一下 Dom 元素的层级。
这种操作无论同时执行多少次都不会有太大的影响,所以对于这一类我们就随他去吧,毕竟线上也不会执行多次。

5)日志 log 上报类
useEffect(() => {
  reportLog({ name: 'viewCount' });
}, []);

对于日志上报类,其实也可以算是无须清理类,但是又有点特殊。
因为,对于日志类,首先在开发环境中我们其实是无须进行上报的,毕竟这种日志打上去也没啥用。
当然,如果是要对上报日志本身这个进行调试等必须上报的情形,这种也有三种应对方式:

方式一,在本地开发环境使用 console.log 来代替 reportLog。
方式二,取消掉严格模式(StrictMode) 方式三,构建一个 production
版本启动,或者将其部署到 QA 环境,部署的时候,指定 production 模式。

借鉴链接:大神地址:epoos文章来源地址https://www.toymoban.com/news/detail-440299.html

到了这里,关于React18的useEffect会执行两次的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • React Hooks 源码解析:useEffect

    React Hooks 源码解析(4):useEffect React 源码版本: v16.11.0 源码注释笔记:airingursb/react 1.1 为什么要有 useEffect 我们在前文中说到 React Hooks 使得 Functional Component 拥有 Class Component 的特性,其主要动机包括: 在组件之间复用状态逻辑很难 复杂组件变得难以理解 难以理解的 class 对

    2024年01月21日
    浏览(41)
  • React - useEffect函数的理解和使用

    我们知道,react 的函数组件里面没有生命周期的,也没有 state,没有 state 可以用 useState 来替代,那么生命周期呢? useEffect 是 react v16.8 新引入的特性。我们可以把 useEffect hook 看作是componentDidMount、componentDidUpdate、componentWillUnmounrt三个生命周期的组合。可以让你在函数式组件

    2024年02月13日
    浏览(35)
  • 【React 】useLayoutEffect 和 useEffect的区别

    useLayoutEffect 和 useEffect 是React中常用的两个Hook,它们的主要区别在于触发时机。 useEffect 会在渲染完成后异步执行,不会阻塞浏览器的绘制操作。它适用于需要在组件渲染后执行副作用的情况,例如数据的获取、订阅事件等。它不会阻止屏幕更新,因此可能会导致渲染的一次

    2024年02月09日
    浏览(40)
  • 通过chatgpt 学习React的useEffect

    定义: useEffect 是 React 中的一个 Hook,它用于处理函数组件中的副作用操作。副作用操作可以包括数据获取、订阅事件、定时器等。 useEffect 接受两个参数:第一个参数是一个回调函数,用于执行副作用操作;第二个参数是一个数组,用于指定依赖项,只有当依赖项发生变化时

    2024年02月10日
    浏览(36)
  • 理解 React 中的 useEffect、useMemo 与 useCallback

    先理解 useEffect 有助于学习 useMemo 和 useCallback。因为 useMemo 和 useCallback 的实现实际上都是基于 useEffect 的。 useEffect 是 React 中的一个很重要的 Hook,用于执行副作用操作。什么是副作用?简单来说,就是那些会改变函数外部变量或有外部可观察影响的操作。useEffect 允许你在函

    2024年02月03日
    浏览(41)
  • React hooks之useEffect、useMemo优化技巧

    useEffect使用JSON.stringfy进行过滤,避免重复执行 将数组直接放入依赖数组可能不会按预期工作,因为数组比较是基于引用而不是内容。也就是说,如果数组引用没有变,即使数组内容发生了变化,副作用也不会重新运行。或者数组内饿哦那个没有改变但是引用却发生变化时,

    2024年02月12日
    浏览(46)
  • 前端react入门day04-useEffect与Hook函数

    (创作不易,感谢有你,你的支持,就是我前行的最大动力,如果看完对你有帮助,请留下您的足迹) 目录 useEffect 的使用 useEffect 的概念理解 useEffect 依赖项参数说明 useEffect — 清除副作用 自定义Hook实现 React Hooks使用规则 useEffect是一个React Hook函数,用于在React组件中 创建不

    2024年01月22日
    浏览(54)
  • React钩子函数之useEffect,useLayoutEffect与useInsertionEffect的区别

    React钩子函数在React开发中扮演着非常重要的角色。其中,useEffect、useLayoutEffect和useInsertionEffect是三个常用的钩子函数,它们的作用虽然有些相似,但是也存在一些区别。在本文中,我们将详细介绍这三个钩子函数的区别,以及它们在React开发中的应用。 首先,我们来了解一下

    2024年02月11日
    浏览(38)
  • React hooks文档笔记(五)useEffect——解决异步操作竞争问题

    非bug,重新安装组件仅在开发过程中发生,帮助找到需要清理的效果。在生产环境中只会加载一次。 React 将在 Effect 下次运行之前以及卸载期间调用您的清理函数。 return () = {}; 在开发中, Effect call addEventListener() ,然后立即call removeEventListener() ,然后再次cal laddEventListener()

    2024年02月11日
    浏览(44)
  • react useState useEffect useMemo实际业务场景中的使用

    下面的代码实现了上面图片的功能

    2024年02月16日
    浏览(42)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包