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 吗?

相关推荐
gyx_这个杀手不太冷静13 分钟前
OpenCode 深度解析:架构设计、工具链集成与工程化实践
前端·架构·ai编程
m0_4592524632 分钟前
fastadmin动态渲染统计信息
开发语言·前端·javascript·php
该怎么办呢44 分钟前
Source/Core/Matrix4.js
前端·javascript
小江的记录本44 分钟前
【MyBatis-Plus】Spring Boot + MyBatis-Plus 进行各种数据库操作(附完整 CRUD 项目代码示例)
java·前端·数据库·spring boot·后端·sql·mybatis
于慨2 小时前
Capacitor
前端
该怎么办呢2 小时前
Source/Core/Event.js
开发语言·javascript·ecmascript·cesium
IT凝冬2 小时前
liunx 的 centos7 安装ngin
前端
赵锦川2 小时前
大屏比例缩放
前端·javascript·html
该怎么办呢2 小时前
Source/Core/DeveloperError.js
开发语言·javascript·ecmascript
于慨2 小时前
tauri
java·服务器·前端