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

相关推荐
Ulyanov2 小时前
Python射击游戏开发实战:从系统架构到高级编程技巧
开发语言·前端·python·系统架构·tkinter·gui开发
华仔啊2 小时前
这 10 个 Vue3 性能优化技巧很实用,但很多项目都没用上
前端·vue.js
手握风云-2 小时前
JavaEE 进阶第九期:Spring MVC - Web开发的“交通枢纽”(三)
前端·spring·java-ee
怕浪猫2 小时前
React从入门到出门第九章 资源加载新特性Suspense 原生协调原理与实战
javascript·react.js·前端框架
天问一2 小时前
Cesium 处理屏幕空间事件(鼠标点击、移动、滚轮)的示例
前端·javascript
@PHARAOH2 小时前
WHAT - Vercel react-best-practices 系列(五)
前端·react.js·前端框架
bjzhang752 小时前
使用 HTML + JavaScript 实现多会议室甘特视图管理系统
前端·javascript·html
qiqiliuwu2 小时前
VUE3+TS+ElementUI项目中监测页面滚动scroll事件以及滚动高度不生效问题的解决方案(window.addEventListener)
前端·javascript·elementui·typescript·vue
LawrenceLan2 小时前
16.Flutter 零基础入门(十六):Widget 基础概念与第一个 Flutter 页面
开发语言·前端·flutter·dart