React 简析useEffect return执行时机

import React, { useState } from "react";
import ReactDOM from "react-dom";

function App() {
  const [n, setN] = useState(0);
  const onClick = () => {
    setN(n + 1);
  };
  React.useEffect(() => {
    console.log("App");
    return () => {
      console.log("App挂了");
    };
  });
  return (
    <div className="App">
      <h1>n: {n}</h1>
      <button onClick={onClick}>+1</button>
      {/* {n % 2 === 0 ? <B /> : ""} */}
      <B />
    </div>
  );
}

function B() {
  const [m, setM] = useState(0);
  const onClick = () => {
    setM(m + 1);
  };
  React.useEffect(() => {
    console.log("B");
    return () => {
      console.log("B挂了");
    };
  });
  return (
    <div>
      B组件
      <h1>m: {m}</h1>
      <button onClick={onClick}>+1</button>
    </div>
  );
}

const rootElement = document.getElementById("root");
ReactDOM.render(<App />, rootElement);

注意点:useEffect 是在render结束之后才执行的。

组件 App 首次渲染后,先执行 console.log("B"); 再执行 console.log("App")

当执行 n + 1 之后,先执行 console.log("B挂了"),再执行执行 console.log("B"), 再执行 console.log("App挂了"), 最后执行console.log("App"), 程序结束。

当执行 m + 1 之后,先执行console.log("B挂了"),再执行console.log("B"), 程序结束。

当组件 App内,使用 useState 创建的变量,发生变化时,会造成重新render,也就导致原组件(包含子组件)的销毁,以及新组件(包含子组件)的诞生。

可以得出,每次重新渲染,都会导致原组件(包含子组件)的销毁,以及新组件(包含子组件)的诞生

结论

1、首先渲染,并不会执行useEffect中的 return

2、变量修改后,导致的重新render,会先执行 useEffect 中的 return,再执行useEffect内除了return部分代码。

3、return 内的回调,可以用来清理遗留垃圾,比如订阅或计时器 ID 等占用资源的东西。

文章参考:React 简析useEffect return执行时机 - 知乎

React的`useEffect` Hook中,返回一个函数(通常称为清理函数)有以下几个关键点[^2]: 1. **默认行为**:当`useEffect`执行完毕后,如果指定了依赖数组(`dependencies`),那么在组件重新渲染之前,React会调用这个清理函数以释放之前可能创建的资源。这是为了防止内存泄漏。 2. **依赖更新**:即使依赖数组未改变,当组件卸载(如通过导航离开页面)时,`useEffect`的返回函数也会被执行一次,这被称为“卸载清理”。在这个阶段,清理函数用于清理那些只在当前组件存在期间使用的临时资源。 3. **依赖变更**:如果依赖数组中有变化,`useEffect`会在下次组件渲染时再次执行,这时它的返回函数也会被执行。这意味着你可以在这里添加额外的清理操作,比如取消订阅事件或关闭网络连接等。 以下是一个简单的示例: ```jsx import { useEffect, useState } from 'react'; function MyComponent() { const [count, setCount] = useState(0); useEffect(() => { console.log('effect', count); return () => { console.log('effect 清理', count); // 当依赖项count变化或组件卸载时都会执行此清理函数 }; }, [count]); // 指定count作为依赖,因此每次count变化时都会执行 return ( <div> <button onClick={() => setCount(pre => pre + 1)}>count + 1</button> </div> ); } ``` 当你点击按钮增加`count`时,可以看到`effect`和`effect 清理`交替打印,反映出每次依赖变化后的执行情况。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值