react中useEffect和useLayoutEffect的区别

文章详细阐述了ReactHooks中的useEffect和useLayoutEffect的执行顺序和特点。useLayoutEffect在DOM更新后但页面渲染前执行,常用于处理布局改变,而useEffect则在页面渲染后调用,适合不涉及布局的副作用操作。使用useLayoutEffect可能会阻塞页面显示,因此通常优先选择useEffect以优化用户体验。
摘要由CSDN通过智能技术生成

布局上

  • useEffect在浏览器渲染完成后执行
  • useLayoutEffect在DOM更新后执行

特点

  • useLayoutEffect 总是比 useEffect 先执行;useLayoutEffect与componentDidMount、componentDidUpdate调用时机相同,都是在DOM更新后,页面渲染前调用;useEffect在页面渲染后调用
  • 使用 useLayoutEffect 时,里面的作用最好改变布局 ,否则会占用等待时间,useLayoutEffect中进行DOM操作,这些DOM修改会与react做出的修改一起被一次性的渲染到页面上,只会产生一次回流、重绘,防止页面抖动

经验

  • 为了用户体验,优先使用 useEffect(优先渲染),因为大部分时候,我们不会去改变DOM
  • useLayoutEffect 会影响用户看到画面变化的时间

例子

  • useEffect
    import React, { useEffect, useState, useLayoutEffect, useRef } from 'react';
    import { render } from 'react-dom';
    
    function App() {
      const [count, setCount] = useState(0);
      
      useEffect(() => {
        if (count === 0) {
          const randomNum = 10 + Math.random()*200
          setCount(10 + Math.random()*200);
        }
      }, [count]);
    
      return (
          <div onClick={() => setCount(0)}>{count}</div>
      );
    }
    
    render(<App />, document.getElementById('root'));

    在线进行试验:React (forked) - StackBlitz

  • useLayoutEffect

    import React, { useEffect, useState, useLayoutEffect, useRef } from 'react';
    import { render } from 'react-dom';
    
    function App() {
      const [count, setCount] = useState(0);
      
      useLayoutEffect(() => {
        if (count === 0) {
          const randomNum = 10 + Math.random()*200
          setCount(10 + Math.random()*200);
        }
      }, [count]);
    
      return (
          <div onClick={() => setCount(0)}>{count}</div>
      );
    }
    
    render(<App />, document.getElementById('root'));

    在线验证:React (forked) - StackBlitz

ReactuseEffectuseLayoutEffect都是用于处理组件副作用的Hook。它们都可以在组件渲染后执行一些操作,比如获取数据、修改DOM等。它们的主要区别在于执行时机和阻塞UI渲染的能力。具体来说,useEffect会在浏览器渲染完成后异步执行,而useLayoutEffect会在浏览器渲染前同步执行,有阻塞UI渲染的能力。因此,如果要修改DOM并且希望能够同步更新UI,则应该使用useLayoutEffect;否则,使用useEffect即可。 举个例子,假设你需要在组件渲染后通过API获取数据并更新组件状态。则你可以这样使用useEffect: ```javascript import { useState, useEffect } from 'react'; function MyComponent() { const [data, setData] = useState(null); useEffect(() => { fetchData().then((res) => setData(res)); }, []); return ( <div> {data ? ( <ul> {data.map((item) => ( <li key={item.id}>{item.name}</li> ))} </ul> ) : ( <p>Loading...</p> )} </div> ); } ``` 这里的useEffect会在组件渲染后异步执行fetchData函数并更新组件状态。而如果你需要在DOM更新之前计算一些值并更新组件状态,则可以使用useLayoutEffect: ```javascript import { useState, useLayoutEffect } from 'react'; function MyComponent() { const [width, setWidth] = useState(null); useLayoutEffect(() => { function updateWidth() { setWidth(window.innerWidth); } window.addEventListener('resize', updateWidth); updateWidth(); return () => window.removeEventListener('resize', updateWidth); }, []); return <p>Window width: {width}px</p>; } ``` 这里的useLayoutEffect会在组件渲染前同步执行updateWidth函数并更新组件状态,因此可以实时获取窗口宽度并渲染到UI
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值