React 之 useEffect

useEffect 的作用

useEffect 是 React Hooks 中用于处理副作用的钩子函数。副作用是指那些不直接与组件渲染相关,但需要在组件生命周期中执行的操作,例如数据获取、订阅事件、手动修改 DOM 等。

如何理解副作用

在 React 中,组件的核心逻辑是渲染 UI,但某些操作(如 API 调用、定时器、事件监听)会影响外部状态或执行非渲染相关的任务,这些操作被称为"副作用"。

useEffect 的基本用法

jsx 复制代码
useEffect(() => {
  // 副作用逻辑(如数据获取、订阅事件)
  return () => {
    // 清理逻辑(如取消订阅、清除定时器)
  };
}, [dependencies]);
  • 第一个参数(函数):定义副作用的执行逻辑。
  • 第二个参数(依赖数组) :控制副作用何时重新执行。如果依赖项变化,副作用会重新运行;若为空数组 [],则仅在组件挂载和卸载时执行。
  • 清理函数(return):在组件卸载或依赖项变化前执行清理操作(如取消订阅)。

常见用例

数据获取
jsx 复制代码
useEffect(() => {
  const fetchData = async () => {
    const response = await fetch('https://api.example.com/data');
    const data = await response.json();
    setData(data);
  };
  fetchData();
}, []); // 空数组表示仅在组件挂载时执行
事件监听
jsx 复制代码
useEffect(() => {
  const handleResize = () => console.log('Window resized');
  window.addEventListener('resize', handleResize);
  return () => window.removeEventListener('resize', handleResize);
}, []); // 清理函数移除监听
依赖项控制
jsx 复制代码
useEffect(() => {
  console.log('Count updated:', count);
}, [count]); // 仅在 count 变化时执行

关键点

  • 避免无限循环:确保依赖项正确,避免频繁触发副作用。
  • 清理资源:防止内存泄漏(如定时器、订阅未清除)。
  • 按需执行:通过依赖项优化性能,减少不必要的副作用调用。

具体实例解析:

以事件监听为实例,代码如下:

javascript 复制代码
useEffect(() => {
  const handleResize = () => console.log('Window resized');
  window.addEventListener('resize', handleResize);
  return () => window.removeEventListener('resize', handleResize);
}, []); // 清理函数移除监听
运行顺序解析
  1. 组件挂载阶段

    当组件首次渲染时,useEffect 的回调函数会立即执行。代码中的箭头函数 () => { ... } 被调用,完成以下操作:

    • 定义 handleResize 函数。
    • 调用 window.addEventListener('resize', handleResize) 注册事件监听器。
  2. 依赖数组的作用

    依赖数组 [] 表示该 useEffect 仅在组件挂载时执行一次。若依赖数组包含变量(如 [someVar]),则变量变化时会重新执行回调函数及其清理函数。

  3. 清理函数执行时机

    清理函数 return () => { ... } 会在以下两种情况下执行:

    • 组件卸载时(如页面导航或组件被移除)。
    • 依赖项变化导致 useEffect 重新执行前(先执行清理函数,再运行新回调)。
具体流程示例
  • 首次渲染handleResize 定义 → addEventListener 调用 → 监听器生效。
  • 重新渲染(依赖不变):无操作(因依赖数组为空,回调不会重复执行)。
  • 组件卸载 :清理函数 removeEventListener 被调用,移除监听器。
注意事项
  • 清理函数的作用是避免内存泄漏,确保监听器随组件生命周期正确解除绑定。
  • 若依赖数组非空,清理函数会在每次依赖变化时优先执行。
相关推荐
十月的皮皮12 分钟前
C语言学习笔记20260606- 求月份天数三种写法
c语言·笔记·学习
naildingding22 分钟前
3-ts接口 Interface
前端·typescript
mONESY22 分钟前
JavaScript 栈、队列、数组与链表核心知识点总结
javascript·面试
小小前端仔LC29 分钟前
Node.js + LangChain + React:搭建个人知识库(六)- “吃什么”项目实战:从700+菜谱入库到Taro H5端JSON渲染
前端·后端
ZengLiangYi29 分钟前
TypeScript 项目配置:tsconfig、ESM、路径别名
javascript·typescript·aigc
马士兵教育29 分钟前
Java还有前景吗?Java+AI大模型学习路线及项目?
java·人工智能·python·学习·机器学习
晓13131 小时前
【Cocos Creator 3.x】篇——第二章 入门
前端·javascript·游戏引擎
想要成为糕糕手1 小时前
前端必修课:JavaScript 数组与数据结构底层逻辑全解析
javascript·数据结构·面试
程序员黑豆1 小时前
AI全栈开发之Java:怎么配置Java环境变量
前端·后端·ai编程
xiaofeichaichai1 小时前
React Hooks
前端·javascript·react.js