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

相关推荐
xiaotao1312 小时前
第九章:Vite API 参考手册
前端·vite·前端打包
午安~婉2 小时前
Electron桌面应用聊天(续)
前端·javascript·electron
彧翎Pro2 小时前
基于 RO1 noetic 配置 robosense Helios 32(速腾) & xsense mti 300
前端·jvm
小码哥_常2 小时前
解锁系统设置新姿势:Activity嵌入全解析
前端
之歆3 小时前
前端存储方案对比:Cookie-Session-LocalStorage-IndexedDB
前端
哟哟耶耶3 小时前
vue3-单文件组件css功能(:deep,:slotted,:global,useCssModule,v-bind)
前端·javascript·css
是罐装可乐3 小时前
深入理解“句柄(Handle)“:从浏览器安全到文件系统访问
前端·javascript·安全
华科易迅3 小时前
Vue如何集成封装Axios
前端·javascript·vue.js
康一夏3 小时前
Next.js 13变化有多大?
前端·react·nextjs
糖炒栗子03263 小时前
前端项目标准环境搭建与启动
前端