探讨useEffect和useLayoutEffect的区别

1.背景

面试官问到:请说明下useEffectuseLayoutEffect的区别。

我:一般开发场景都是用的useEffect,react官方文档上说,两者的使用场景稍微有些不同,但是还是推荐尽量使用useEffect

面试官:不够细节,再说点,摩多摩多。

我:...

2.官网介绍

对比 useEffect useLayoutEffect
简介 useEffect 是一个 React Hook,它允许你 将组件与外部系统同步 useLayoutEffectuseEffect 的一个版本,在浏览器重新绘制屏幕之前触发。
aip useEffect(setup, dependencies?) useLayoutEffect(setup, dependencies?)
返回值 undefined undefined

3. 对比总结

3.1 useEffect

3.1.1 依赖项对effect执行的影响

  • 无依赖项:没有依赖项数组:每次重新渲染后重新运行!

  • 错误依赖项:

    • 依赖项每次渲染都不同:指定了依赖项数组,你的 Effect 仍循环地重新运行

      • => 解决:手动打印依赖项数组进行排查:console.log([dep1, dep2]);

3.1.2 运行闪烁

我的 Effect 做了一些视觉相关的事情,在它运行之前我看到了一个闪烁

  • 产生原因:useEffect是异步宏任务,在下一轮事件循环才会执行。而GUI渲染和JS线程是互斥的,也就是说,在Effect 运行前,浏览器就已经进行了屏幕渲染,渲染完毕后,useEffect才会被任务队列取出来,开始执行。

  • 解决方法:

    • 1.在得到正确的数据之前,使用loading
    • 2.使用useLayoutEffect

3.2 useLayoutEffect

3.2.1 依赖项对effect执行的影响

useEffect

3.2.2 运行闪烁

useLayoutEffect是同步任务,也就是说,他会在GUI渲染之前触发。

useLayoutEffect 内部的代码和所有计划的状态更新阻塞了浏览器重新绘制屏幕。如果过度使用,这会使你的应用程序变慢。如果可能的话,尽量选择 useEffect

  • 使用场景:

    • 在浏览器重绘前计算布局(官网以toolTip组件为例)
scss 复制代码
function Tooltip() {
  const ref = useRef(null);
  const [tooltipHeight, setTooltipHeight] = useState(0); // 你还不知道真正的高度
​
  useLayoutEffect(() => {
    const { height } = ref.current.getBoundingClientRect();
    setTooltipHeight(height); // 现在重新渲染,你知道了真实的高度
  }, []);
​
  // ... 在下方的渲染逻辑中使用 tooltipHeight ...
}

4. 一个小demo

准备了一个小demo,可以直观的体现出二者的区别

4.1 部分关键代码:

scss 复制代码
function App() {
  const [num1, setNum1] = useState(0);
  const [num2, setNum2] = useState(0);
  useEffect(() => {
    console.log("b");
    setNum1(num1 + 1);
  }, []);
  useLayoutEffect(() => {
    console.log("a");
    setNum2(num2 + 1);
  }, []);
  useEffect(() => {
    console.log(num1, num2);
  }, [num1, num2]);
  return (
    <div className="App">
      <header className="App-header">
        {num1} -- {num2}
      </header>
    </div>
  );
}

4.2 效果展示:

4.3 devtool打印结果:

浏览器设置如下:

  • chrome devtools性能节流 配置 cpu六倍降速

4.4 结果分析

5. 最后

nextTick一般用于等待dom操作完成之后马上执行:等待弹窗打开后进行数据回显,这一点来说,与useEffect的用途类似。非交互式的场景下,在执行逻辑不太多的情况下,可以使用useLayoutEffect,避免闪烁,也避免页面白屏时间过长。

6. 参考资料

React官方文档

相关推荐
xiao-xiang14 分钟前
jenkins-通过api获取所有job及最新build信息
前端·servlet·jenkins
C语言魔术师31 分钟前
【小游戏篇】三子棋游戏
前端·算法·游戏
匹马夕阳2 小时前
Vue 3中导航守卫(Navigation Guard)结合Axios实现token认证机制
前端·javascript·vue.js
你熬夜了吗?2 小时前
日历热力图,月度数据可视化图表(日活跃图、格子图)vue组件
前端·vue.js·信息可视化
screct_demo3 小时前
詳細講一下在RN(ReactNative)中,6個比較常用的組件以及詳細的用法
javascript·react native·react.js
桂月二二8 小时前
探索前端开发中的 Web Vitals —— 提升用户体验的关键技术
前端·ux
hunter2062069 小时前
ubuntu向一个pc主机通过web发送数据,pc端通过工具直接查看收到的数据
linux·前端·ubuntu
qzhqbb9 小时前
web服务器 网站部署的架构
服务器·前端·架构
刻刻帝的海角9 小时前
CSS 颜色
前端·css
九酒10 小时前
从UI稿到代码优化,看Trae AI 编辑器如何帮助开发者提效
前端·trae