HOW - React 如何在在浏览器绘制之前同步执行 - useLayoutEffect

news2025/3/9 13:40:19

目录

  • useEffect vs useLayoutEffect
    • useEffect
    • useLayoutEffect
    • 主要区别总结
    • 选择建议
    • 注意事项
  • useLayoutEffect 使用示例
    • 测量 DOM 元素的尺寸和位置
      • 示例:自适应弹出框定位
    • 同步更新样式以避免闪烁
      • 示例:根据内容动态调整容器高度
    • 图像或 Canvas 绘制前的准备工作
      • 示例:Canvas 绘制自定义图形

useEffect vs useLayoutEffect

useLayoutEffectuseEffect 都是 React Hooks 中用于处理副作用的钩子,但它们在执行时机和用途上有一些关键区别。

理解这些区别有助于在不同的场景下选择合适的钩子来优化组件的性能和用户体验。

useEffect

  • 执行时机

    • useEffect 的回调函数会在浏览器完成渲染之后异步执行。这意味着它不会阻塞浏览器的绘制过程。
  • 用途

    • 适用于大多数副作用操作,如数据获取、订阅、手动操作 DOM(例如添加事件监听器)、设置定时器等。
    • 因为它在渲染后异步执行,所以不会导致用户界面的阻塞,适合处理不影响当前渲染结果的副作用。
  • 示例

import React, { useEffect } from 'react';

function Example() {
  useEffect(() => {
    // 这里的代码会在组件渲染到屏幕后异步执行
    console.log('useEffect 执行');

    return () => {
      // 清理函数
      console.log('useEffect 清理');
    };
  }, []); // 空依赖数组表示只在组件挂载和卸载时执行

  return <div>Hello World</div>;
}

useLayoutEffect

  • 执行时机

    • useLayoutEffect 的回调函数会在 DOM 更新完成后、浏览器进行绘制之前同步执行。这意味着它会阻塞浏览器的绘制过程,直到回调函数执行完毕。
  • 用途

    • 适用于需要在浏览器绘制之前同步执行的操作,如测量 DOM 元素的尺寸、位置,或者进行需要同步更新样式的操作。
    • 由于它会阻塞渲染,应谨慎使用,避免在高频率更新的场景下使用,以免影响性能。
  • 示例

import React, { useLayoutEffect, useRef, useState } from 'react';

function LayoutEffectExample() {
  const divRef = useRef(null);
  const [width, setWidth] = useState(0);

  useLayoutEffect(() => {
    // 在浏览器绘制之前同步获取 DOM 元素的宽度
    if (divRef.current) {
      const { width } = divRef.current.getBoundingClientRect();
      setWidth(width);
    }
  }, []);

  return (
    <div>
      <div ref={divRef} style={{ width: '50%', border: '1px solid black' }}>
        测量宽度
      </div>
      <p>宽度: {width}px</p>
    </div>
  );
}

主要区别总结

特性useEffectuseLayoutEffect
执行时机渲染后异步执行渲染后、绘制前同步执行
阻塞渲染不阻塞阻塞
适用场景大多数副作用操作,如数据获取、订阅等需要同步操作 DOM 或测量布局的场景
性能影响较低,适合频繁使用的副作用高,需谨慎使用以避免阻塞渲染

选择建议

  • 优先使用 useEffect:在大多数情况下,useEffect 足以满足需求,并且由于其异步执行的特性,不会影响用户界面的渲染性能。

  • 必要时使用 useLayoutEffect:只有在确实需要在浏览器绘制之前同步执行某些操作时,才使用 useLayoutEffect。例如,调整布局、测量元素尺寸等。

注意事项

  • 避免过度使用 useLayoutEffect:由于其会阻塞渲染,频繁或不当使用可能导致页面卡顿或响应缓慢。

  • 清理函数:与 useEffect 一样,useLayoutEffect 也支持返回一个清理函数,用于在组件卸载或依赖项变化时执行清理操作。

通过理解 useEffectuseLayoutEffect 的区别,可以更有效地管理组件的副作用,优化性能,并提升用户体验。

useLayoutEffect 使用示例

测量 DOM 元素的尺寸和位置

在某些情况下,你需要精确知道某个 DOM 元素的实际尺寸(宽度、高度)或在页面中的位置信息,以便根据这些信息进行后续的操作,而这些操作必须在浏览器绘制之前完成,否则可能会导致测量的不准确。

示例:自适应弹出框定位

当你有一个弹出框,需要根据触发它的元素的相对位置来动态定位时,就需要先测量触发元素的尺寸和位置,然后根据这些信息计算出弹出框的合适位置。

import React, { useRef, useState, useLayoutEffect } from 'react';

const PopupExample = () => {
    const triggerRef = useRef(null);
    const popupRef = useRef(null);
    const [position, setPosition] = useState({ top: 0, left: 0 });

    useLayoutEffect(() => {
        if (triggerRef.current && popupRef.current) {
            const triggerRect = triggerRef.current.getBoundingClientRect();
            // 简单示例,将弹出框定位在触发元素的下方
            const newTop = triggerRect.bottom + window.scrollY;
            const newLeft = triggerRect.left + window.scrollX;

            setPosition({ top: newTop, left: newLeft });
        }
    }, []);

    return (
        <div>
            <button ref={triggerRef}>点击显示弹出框</button>
            <div ref={popupRef} style={{ position: 'absolute', top: position.top, left: position.left }}>
                这是一个弹出框
            </div>
        </div>
    );
};

export default PopupExample;

在上述代码中,useLayoutEffect 会在浏览器绘制之前同步执行,确保能够准确获取触发元素的尺寸和位置信息,从而正确地定位弹出框。

同步更新样式以避免闪烁

有时候,你需要根据某些计算结果立即更新元素的样式,而且希望这些更新在浏览器绘制之前完成,以避免出现样式闪烁的问题。

示例:根据内容动态调整容器高度

当容器内的内容动态变化时,你可能希望容器的高度能够立即适应内容的变化,而不让用户看到内容先溢出再调整高度的过程。

import React, { useRef, useState, useLayoutEffect } from 'react';

const DynamicHeightContainer = () => {
    const containerRef = useRef(null);
    const [content, setContent] = useState('初始内容');

    const toggleContent = () => {
        setContent(prevContent => prevContent === '初始内容' ? '这是一段更长更长的内容,用于测试容器高度的动态调整。' : '初始内容');
    };

    useLayoutEffect(() => {
        if (containerRef.current) {
            // 这里可以根据内容做一些样式调整,例如设置高度
            // 示例中简单打印信息,实际应用中可根据需求修改样式
            console.log('根据内容调整容器样式');
        }
    }, [content]);

    return (
        <div>
            <button onClick={toggleContent}>切换内容</button>
            <div ref={containerRef} style={{ border: '1px solid black' }}>
                {content}
            </div>
        </div>
    );
};

export default DynamicHeightContainer;

在这个例子中,useLayoutEffect 确保在浏览器绘制之前根据内容的变化同步更新容器的样式,避免了内容溢出或高度闪烁的问题。

图像或 Canvas 绘制前的准备工作

在进行图像处理或使用 Canvas 进行绘图时,有时需要在绘制之前获取一些必要的信息或进行一些预处理操作,这些操作也需要在浏览器绘制之前完成。

示例:Canvas 绘制自定义图形

在使用 Canvas 绘制复杂图形时,可能需要先测量某些元素的位置或尺寸,然后根据这些信息进行绘制。

import React, { useRef, useLayoutEffect } from 'react';

const CanvasDrawing = () => {
    const canvasRef = useRef(null);

    useLayoutEffect(() => {
        const canvas = canvasRef.current;
        const ctx = canvas.getContext('2d');

        // 假设这里需要根据某个 DOM 元素的位置来绘制图形
        // 先进行相关测量等准备工作(此处简化)

        // 开始绘制
        ctx.fillStyle = 'red';
        ctx.fillRect(10, 10, 50, 50);
    }, []);

    return <canvas ref={canvasRef} width="200" height="200"></canvas>;
};

export default CanvasDrawing;

在上述代码中,useLayoutEffect 可以确保在进行 Canvas 绘制之前完成所有必要的准备工作,保证绘制的准确性和流畅性 。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2311148.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

前端开发10大框架深度解析

摘要 在现代前端开发中&#xff0c;框架的选择对项目的成功至关重要。本文旨在为开发者提供一份全面的前端框架指南&#xff0c;涵盖 React、Vue.js、Angular、Svelte、Ember.js、Preact、Backbone.js、Next.js、Nuxt.js 和 Gatsby。我们将从 简介、优缺点、适用场景 以及 实际…

图像形成与计算机视觉基础

1. 图像形成的基本原理 图像形成是物理世界与传感器&#xff08;如胶片、CCD/CMOS&#xff09;交互的过程&#xff0c;核心是光线的传播与记录。 1.1 直接放置胶片模型 物理原理&#xff1a;物体表面反射的光线直接照射到胶片上&#xff0c;但无任何遮挡或聚焦机制。 问题&a…

Spring Boot 缓存最佳实践:从基础到生产的完整指南

Spring Boot 缓存最佳实践&#xff1a;从基础到生产的完整指南 引言 在现代分布式系统中&#xff0c;缓存是提升系统性能的银弹。Spring Boot 通过 spring-boot-starter-cache​ 模块提供了开箱即用的缓存抽象&#xff0c;但如何根据业务需求实现灵活、可靠的缓存方案&#xf…

Ubuntu20.04双系统安装及软件安装(一):系统安装

Ubuntu20.04双系统安装及软件安装&#xff08;一&#xff09;&#xff1a;系统安装 Ubuntu系统卸载Ubuntu20.04安装BIOS进入系统安装 许久没写博客了&#xff0c;今天开始重新回归了。首先记录我在双系统上重装Ubuntu20.04的安装过程记录以及个人见解。 Ubuntu系统卸载 参考双…

Linux14-io多路复用

UDP:单循环服务器,服务器同一时刻只能响应一个客户端的请求 TCP:并发服务器,服务器同一时刻只能响应多个客户端的请求 一、构建TCP并发服务器 让TCP服务端具备同时响应多个客户端的能力。 1.多进程 资源消耗大,同资源平台下,并发量小。 2.多线程 创建线程、进程,比…

Next.js项目实战-ai助手帮我写文章发布视频第1节(共89节)

&#x1f602;Ai在国内外已经杀疯了&#xff0c;老板要求我们把速度再提升快一些&#xff0c;哪怕是几秒&#xff0c;几百毫秒也行&#xff5e;现在&#xff0c;马上就要&#xff0c;就地就要&#xff0c;只好搬出前端服务端(大保健)&#x1f613;。没错&#xff0c;今天我要分…

探秘Transformer系列之(9)--- 位置编码分类

探秘Transformer系列之&#xff08;9&#xff09;— 位置编码分类 文章目录 探秘Transformer系列之&#xff08;9&#xff09;--- 位置编码分类0x00 概述0x01 区别1.1 从直观角度来看1.2 从模型处理角度来看1.3 优劣 0x02 绝对位置编码2.1 基础方案2.2 训练式2.3 三角函数式2.4…

笔记四:C语言中的文件和文件操作

Faye&#xff1a;只要有正确的伴奏&#xff0c;什么都能变成好旋律。 ---------《寻找天堂》 目录 一、文件介绍 1.1程序文件 1.2 数据文件 1.3 文件名 二、文件的打开和关闭 2.1 文件指针 2.2.文件的打开和关闭 2.3 文件读取结束的判定 三、 文件的顺序读写 3.1 顺序读写…

Zabbix+Deepseek实现AI告警分析(非本地部署大模型版)

目录 前言技术架构DeepSeek API获取1. 注册账号2. 申请API-Key Zabbix告警AI分析 实现1. 创建Scripts2. Scripts关键参数说明3. 需要注意 测试参考链接 前言 最近手伤了&#xff0c;更新频率下降…… 近期在Zabbix社区看到了一篇文章&#xff1a;张世宏老师分享的《Zabbix告警分…

国产NAS系统飞牛云fnOS深度体验:从运维面板到博客生态全打通

文章目录 前言1. 飞牛云本地部署1Panel2. 1Panel功能介绍3. 公网访问1Panel控制面板4. 固定1Panel公网地址5. 1Panel搭建Halo博客6. 公网访问Halo个人博客 前言 嘿&#xff0c;小伙伴们&#xff01;是不是厌倦了服务器管理的繁琐和搭建个人网站的复杂&#xff1f;今天就来一场…

使用QT + 文件IO + 鼠标拖拽事件 + 线程 ,实现大文件的传输

第一题、使用qss&#xff0c;通过线程&#xff0c;使进度条自己动起来 mythread.h #ifndef MYTHREAD_H #define MYTHREAD_H#include <QObject> #include <QThread> #include <QDebug>class mythread : public QThread {Q_OBJECT public:mythread(QObject* …

博查搜索API日调用量突破3000万次,达到Bing API的1/3。

根据第三方机构统计&#xff0c;2024年Bing Search API 全球日均调用量为1.1亿次。截至2025年3月&#xff0c;博查 Search API日均调用量已达到3000万次&#xff08;约为Bing的1/3&#xff09;&#xff0c;承接着国内AI应用60%的联网搜索请求。

[内网安全] Windows 本地认证 — NTLM 哈希和 LM 哈希

关注这个专栏的其他相关笔记&#xff1a;[内网安全] 内网渗透 - 学习手册-CSDN博客 0x01&#xff1a;SAM 文件 & Windows 本地认证流程 0x0101&#xff1a;SAM 文件简介 Windows 本地账户的登录密码是存储在系统本地的 SAM 文件中的&#xff0c;在登录 Windows 的时候&am…

输电线路杆塔倾斜智能监测:守护电网安全的智慧之眼

​ ​2023年夏&#xff0c;某超高压输电线路突发倒塔事故&#xff0c;导致三省市大面积停电&#xff0c;直接经济损失超2.3亿元。事后调查显示&#xff0c;杆塔倾斜角度早已超出安全阈值&#xff0c;但传统巡检未能及时发现。这个刺痛行业的案例&#xff0c;揭开了电力设施监…

探索.NET 10 的新特性,开发效率再升级!

前言 最近&#xff0c;.NET 10 发布啦&#xff0c;作为长期支持&#xff08;LTS&#xff09;版本&#xff0c;接下来的 3 年里它会给开发者们稳稳的幸福。今天咱就来唠唠它都带来了哪些超实用的新特性。可在指定链接下载。 新特性 下面将介绍了.NET 10的新特性&#xff0c;其…

算法·搜索

搜索问题 搜索问题本质也是暴力枚举&#xff0c;一般想到暴力也要想到利用回溯枚举。 排序和组合问题 回溯法 去重问题&#xff1a;定义全局变量visited还是局部变量visited实现去重&#xff1f; 回溯问题 图论中的搜索问题 与一般的搜索问题一致&#xff0c;只不过要多…

《水利水电安全员考试各题型对比分析及应对攻略》

《水利水电安全员考试各题型对比分析及应对攻略》 单选题&#xff1a; 特点&#xff1a;四个选项中只有一个正确答案&#xff0c;相对难度较小。主要考查对基础知识的掌握程度。 应对攻略&#xff1a;认真审题&#xff0c;看清题目要求。对于熟悉的知识点&#xff0c;直接选择…

鸿蒙HarmonyOS-Navagation基本用法

Navagation基本用法 Navigation组件是路由导航的根视图容器&#xff0c;一般作为Page页面的根容器使用&#xff0c;其内部默认包含了标题栏&#xff0c;内容栏和公工具栏&#xff0c;其中内容区默认首页显示导航内容&#xff08;Navigation的子组件&#xff09;或非首页显示&am…

【AI深度学习网络】卷积神经网络(CNN)入门指南:从生物启发的原理到现代架构演进

深度神经网络系列文章 【AI深度学习网络】卷积神经网络&#xff08;CNN&#xff09;入门指南&#xff1a;从生物启发的原理到现代架构演进【AI实践】基于TensorFlow/Keras的CNN&#xff08;卷积神经网络&#xff09;简单实现&#xff1a;手写数字识别的工程实践 引言 在当今…

江科大51单片机笔记【10】蜂鸣器播放提示器音乐(下)

一、蜂鸣器播放提示器 这里我们要用Key&#xff0c;Delay&#xff0c;Nixie模块 并且把Nixie.c函数里的这两句注释&#xff0c;因为之前是动态显示&#xff0c;延时后马上清零&#xff0c;现在是静态显示&#xff0c;所以需要把他注释掉 // Delay(1); // P00x00; 先验…