useEffect和useLayoutEffect区别

官方解释

官方解释,这两个hook基本相同,调用时机不同,请全部使用useEffect,除非遇到bug或者不可解决的问题,再考虑使用useLayoutEffect。还举了个例子,譬如你想测量DOM元素时候,使用useLayoutEffect。个人感觉举例不恰当,测试DOM我也完全可以在useEffect中测量啊。说如果需要在paint前改变DOM,更合适。

我做过测试,譬如一个div尺寸是200 * 200,我想改成100 * 100,如果写在useEffect中,确实会造成页面抖动,写在useLayoutEffect中可以避免。

官方解释链接

官方解释

redux-react-hook 中的妙用

redux-react-hook库中有段代码使用了useLayoutEffect,用来避免组件render两次。
这里的useIsomorphicLayoutEffect就是useLayoutEffect(因为库要区分是浏览器还是SSR,所以上面做了处理)

    // We use useLayoutEffect to render once if we have multiple useMappedState. 
    // We need to update lastStateRef synchronously after rendering component,
    // With useEffect we would have:
    // 1) dispatch action
    // 2) call subscription cb in useMappedState1, call forceUpdate
    // 3) rerender component
    // 4) call useMappedState1 and useMappedState2 code
    // 5) calc new derivedState in useMappedState2, schedule updating lastStateRef, return new state, render component
    // 6) call subscription cb in useMappedState2, check if lastStateRef !== newDerivedState, call forceUpdate, rerender.
    // 7) update lastStateRef - it's too late, we already made one unnecessary render
    useIsomorphicLayoutEffect(() => {
      lastStateRef.current = derivedState;
      memoizedMapStateRef.current = memoizedMapState;
    });

看得很懵逼,讲了如果用useEffect会带来什么问题,我模拟了很久终于模拟出来作者描述的问题(意图好猜,模拟时候有个细节很难处理)

模拟场景简化

场景

我有一个数据store(对redux的store),一个组件App,组件中使用了useA和useB两个自定义hook(这对应两次调用redux-react-hook的useMappedState)。

当我一个操作,改变store时候,去调用订阅者即A和B,A和B改变会触发App重新render。这里有个问题,A和B都是订阅者,会触发两次App重新render,作者想避免,所以会在use的时候做下处理,使用useEffect的话,会出现bug,无法如愿,下面就来模拟这个过程。

代码实现

function App() {
  console.log('%c App render--start-->', 'color:blue')
  const a = useA();
  const b = useB();
  
  function doSomething() {
    // dispatch();
    setTimeout(dispatch, 0)
  }
  console.log('%c App render--end-->', 'color:red')
  return (
    <div>
      <p>a: {a}</p>
      <p>b: {b}</p>
      <p><button onClick={doSomething}>dispatch</button></p>
      
    </div>
  )
}
function useA() {
  console.log('---a--hook-->')
  const [trigger, setTrigger] = useState(0);
  useEffect(() => {
    console.log('--useA--useEffect-->')
    memoStore = store;
  });
  useEffect(() => {
    const fn = subsriber(() => {
      console.log('--useA--注册函数--->', memoStore, store);
      if(store !== memoStore) {
        setTrigger(Math.random())
      }
    });
    return () => unSubsriber(fn);
  }, []);
  return store;
}
function useB() {
  console.log('---b--hook-->')
  const [trigger, setTrigger] = useState(0);
  useEffect(() => {
    console.log('--useA--useEffect-->')
    memoStore = store
  });
  useEffect(() => {
    const fn = subsriber(() => {
      console.log('--useB--注册函数--->', memoStore, store);
      if(store !== memoStore) {
        setTrigger(Math.random())
      }
    });
    return () => unSubsriber(fn);
  }, []);
  return store;
}

简化的redux:

let store = 6;
let memoStore = 6;
const newStore = 8;

const subsriberList = new Set();
function subsriber(fn) {
  subsriberList.add(fn);
  return fn;
}
function unSubsriber(fn) {
  subsriberList.delete(fn)
}
function dispatch() {
  memoStore = store;
  store = newStore;
  subsriberList.forEach(fn => fn())
}

这里有一个非常重要的关键点,就是App组件中的doSometing中,dispatch一定要写在setTimeout中,否则react自动帮你优化了,模拟不出来想要的场景。

分析

点击按钮时候,改变了store: 6 -> 8,触发了订阅者自定义hook A和B的订阅事件。按理会触发两次App render,但是我们做了优化,在useA和useB的时候,会用新状态去覆盖旧状态,然后在订阅事件中,会对比新老状态,一致的话,就不去触发自定义hook改变了,也就不会触发App render了。
但是使用effect的话,实际执行过程是这样的:


使用effect的代码执行流程
控制台

可以看到,App依旧render了两次,其中主要问题就出在useEffect注册的函数在什么时候执行,从流程图中可以看到,其不是在App组件树 render结束后立即执行的(我也不知道什么时候执行,还请哪位大佬指点),js会继续执行后面的代码(B的订阅),这个时候old=new还没有执行,所以依旧触发了第二次App组件render。

更改useEffect为useLayoutEffect

useA

...
  useLayoutEffect(() => {
    console.log('--useA--useLayoutEffect-->')
    memoStore = store;
  });
...

useB

...
  useLayoutEffect(() => {
    console.log('--useA--useLayoutEffect-->')
    memoStore = store
  });
...
useLayoutEffect执行流程
控制台

可以看见关键点是,layoutEffect队列在组件树render结束后,会立刻同步执行(个人感觉是的),所以在第一次App render结束后,old和new就相同了,在执行B订阅时候,就会根据条件,不再触发App render了。

总结

// 一定要加setTimeout模拟异步操作,否则实验不出来上面的流程的
  setTimeout(()=>{
    renderApp1(); // 一些会条件性触发组件重新render的代码
    exeLayoutEffectList(); // 组件树构建完毕,会同步执行useLayoutEffect中的代码
    code1(); // 一些js代码
    code2(); // 一些js代码
    // 所有代码都执行完毕后,浏览器渲染结束后,会调用useEffect中的代码
    // 或者接到下一次组件刷新(re-render)指令,会将上一次effect队列执行完毕。我根据试验猜的
    exeEffectList(); 
    renderApp2(); // 一些会条件性触发组件重新render的代码
  }, 0)

主要就是effect和layoutEffect队列的执行阶段,layout会在组件树构建完毕或者刷新完毕后同步立刻执行。effect会等其他js代码执行完毕后执行(或者遇到下一次刷新任务前)

回过头再看react关于useLayoutEffect的官方文档:

The signature is identical to useEffect, but it fires synchronously after all DOM mutations. Use this to read layout from the DOM and synchronously re-render. Updates scheduled inside useLayoutEffect will be flushed synchronously, before the browser has a chance to paint.
Prefer the standard useEffect when possible to avoid blocking visual updates.

  • 和useEffect相同,是指他们都在组件树构建完毕之后执行的
  • 但是useLayout是在DOM突变之后立即执行的,突变是指什么?是指类似组件构建完毕之后,appendChild(reactTree)这种操作吗?
  • 可以肯定的是,是在组件树构建完毕后同步执行,之后才会去执行后面的js代码
  • 使用他来读取DOM布局尺寸,我倒感觉应该是写成设定DOM布局尺寸,这样可以防抖动,同步读取DOM布局尺寸想不懂有什么用
  • useLayoutEffect队列中的任务,会在浏览器paint之前执行(可以用来防抖)
  • 尽可能使用useEffect来避免阻塞视觉更新(见上条,阻碍paint)

吐槽

英语太差,好多概念模模糊糊的,但是好像看过国外文章,也有吐槽react的几个概念含糊不清的。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,686评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,668评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,160评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,736评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,847评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,043评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,129评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,872评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,318评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,645评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,777评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,861评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,589评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,687评论 2 351

推荐阅读更多精彩内容

  • 原教程内容详见精益 React 学习指南,这只是我在学习过程中的一些阅读笔记,个人觉得该教程讲解深入浅出,比目前大...
    leonaxiong阅读 2,813评论 1 18
  • 40、React 什么是React?React 是一个用于构建用户界面的框架(采用的是MVC模式):集中处理VIE...
    萌妹撒阅读 1,005评论 0 1
  • 3. JSX JSX是对JavaScript语言的一个扩展语法, 用于生产React“元素”,建议在描述UI的时候...
    pixels阅读 2,810评论 0 24
  • [toc] REACT react :1.用来构建用户界面的 JAVASCRIPT 库2.react 专注于视图层...
    拨开云雾0521阅读 1,432评论 0 1
  • 【日精进打卡第5天】 【知~学习】诵读 《六项精进》1遍 累计5遍 《大学》1遍 累计5遍 • 【经典名句分享】 ...
    不要有感性的烦恼阅读 89评论 0 0