useeffect和uselayouteffect

在 React 中,useEffectuseLayoutEffect 的核心区别在于 它们在浏览器渲染流程中的触发时机

简单来说:useEffect异步 执行的,不会阻塞屏幕绘制;而 useLayoutEffect同步执行的,会在屏幕绘制之前完成。


1. 执行时机对比

  • useEffect (最常用)
  1. 状态更新,组件重新渲染。
  2. 浏览器将组件的变更绘制到屏幕上。
  3. 绘制完成后useEffect 的回调函数异步执行。
  • 优点: 不会阻塞用户界面,性能更好。

  • useLayoutEffect (特殊用途)

  1. 状态更新,组件重新渲染。
  2. React 计算出 DOM 的变化。
  3. 在浏览器绘制屏幕之前useLayoutEffect 同步执行。
  4. 浏览器将最终结果绘制到屏幕。
  • 优点: 可以防止用户看到视觉上的"闪烁"或抖动。

2. 核心区别摘要

特性 useEffect useLayoutEffect
执行时机 屏幕绘制 (After paint) 屏幕绘制 (Before paint)
执行方式 异步 (Asynchronous) 同步 (Synchronous)
性能影响 低,不阻塞渲染 高,可能会阻塞页面显示
典型场景 数据获取、订阅、日志记录 读取布局、测量 DOM、手动修改 DOM

3. 什么时候用 useLayoutEffect?

你几乎应该总是首选 useEffect。只有当你遇到以下情况时才考虑 useLayoutEffect

场景:避免界面闪烁

假设你需要根据一个按钮的位置来定位一个弹窗。

  • 如果用 useEffect:浏览器先画出位置不对的弹窗,然后 useEffect 运行并修正位置,用户会看到弹窗"跳"了一下。
  • 如果用 useLayoutEffect:React 会在浏览器画出弹窗前就把位置算好,用户直接看到位置正确的弹窗。

4. 代码演示

javascript 复制代码
import React, { useState, useLayoutEffect, useEffect, useRef } from 'react';

function ComparisonExample() {
  const [value, setValue] = useState(0);
  const divRef = useRef();

  useLayoutEffect(() => {
    // 这里的代码会在屏幕更新前执行
    if (value === 0) {
      // 假设我们要强制把 0 变成一个随机大数
      setValue(10 + Math.random() * 200);
    }
  }, [value]);

  return (
    <div ref={divRef} onClick={() => setValue(0)}>
      Value is: {value} (点击我重置为0)
    </div>
  );
}

注意: 在服务器端渲染 (SSR) 时,useLayoutEffect 会引发警告,因为它要求同步访问浏览器 API。这种情况下通常建议使用 useEffect 或进行环境判断。


你想针对某个具体的业务场景(比如动画或复杂的 DOM 测量)看看该如何选择这两个 Hook 吗?

相关推荐
掘金安东尼7 小时前
纯 CSS 实现弹性文字效果
前端·css
牛奶7 小时前
Vue 基础理论 & API 使用
前端·vue.js·面试
牛奶7 小时前
Vue 底层原理 & 新特性
前端·vue.js·面试
anOnion8 小时前
构建无障碍组件之Radio group pattern
前端·html·交互设计
pe7er8 小时前
状态提升:前端开发中的状态管理的设计思想
前端·vue.js·react.js
SoaringHeart9 小时前
Flutter调试组件:打印任意组件尺寸位置信息 NRenderBox
前端·flutter
晚风予星9 小时前
Ant Design Token Lens 迎来了全面升级!支持在 .tsx 或 .ts 文件中直接使用 Design Token
前端·react.js·visual studio code
sunny_10 小时前
⚡️ vite-plugin-oxc:从 Babel 到 Oxc,我为 Vite 写了一个高性能编译插件
前端·webpack·架构
GIS之路10 小时前
ArcPy 开发环境搭建
前端
林小帅11 小时前
【笔记】OpenClaw 架构浅析
前端·agent