suspense 和 lazy 配合实现懒加载的原理浅析

先看下面这段代码

javascript 复制代码
// lazy.js
function Fn() {
  return <div className="App">
    函数组件
  </div>
}

// app.js
// 添加一个固定的延迟时间,以便你可以看到加载状态
function delayForDemo(promise) {
  return new Promise(resolve => {
    setTimeout(resolve, 5000);
  }).then(() => promise);
}

const LazyFn = lazy(() => delayForDemo(import('./lazy.js')));
function SuspendedApp() {
  return (
    <Suspense fallback={<div>加载中。。。</div>}>
      <LazyFn />
    </Suspense>
  )
}


const root = ReactDOM.createRoot(document.getElementById('app'));

root.render(
  <SuspendedApp />
);

这段代码模拟了懒加载的过程,运行之后,屏幕上会先展示 Suspense 的 fallback ------ 加载中。。。 过 5 秒钟加载到组件后会自动重新渲染屏幕上出现 函数组件。。。。那这个 React 是怎么实现的呢,下面我们来看看实现的流程。

实现流程

这个流程发生在 Render 阶段。

  1. 从根节点开始挨个执行 beginWork 方法(不太清楚的话可以看看Render流程
  2. 执行到 Suspense 的 beginWork 时,会通过 Didcaptrue 来判断是否展示 fallback 内容,第一次执行默认不展示 fallback,走正常渲染子节点
  3. 正常渲染 lazy 节点,结果返回一个 Promise,Promise 肯定没办法渲染啊,所以把这个Promise作为异常抛出
  4. 异常被捕获,发现异常是一个 Promise,为这个 Promise.then 传入了一个触发更新的方法,并找到距离这个 lazy 组件最近的 Suspense 组件并标记 shouldCapture 为 true,后续处理后标记 DidCaptrue 为 true
  5. 重新执行 Suspense 的 beginWork 方法,此时第二次执行,Didcaptrue 为 true ,展示 fallback 内容,同时把 Didcaptrue 置为 false。
  6. 懒加载的组件加载完成,触发更新,和步骤 1 一样
  7. 执行 Suspense 的 beginWork 方法,DidCaptrue 为 false,正常展示子节点
  8. 渲染 lazy 节点,此时拿到了组件,正常渲染组件

以上就是主要的实现流程(当然实际会比上面介绍的复杂一点)。

源码环节

先是 Suspense 执行 beginWork

这里就是通过 Didcaptrue 来判断 showFallback 的值的。 第一次执行 showFallback 为 false ,因此先正常展示子节点

之后会处理 OffscreenComponent, 完了才会处理 lazy 组件,这个方法里最关键的就是 init 方法 也是在 init 方法里面抛出了异常

接下来是捕获异常,先是被beginWork 捕获,再抛出,最终被 renderRootSync 函数的错误处理捕获处理

接下来重新处理 Suspense,此时showFallback 为 true,同时处理了子节点和 Fallback 节点,最终返回Fallback节点,并渲染

接下来Promise被解决

再次处理 Suspense,更新子节点

然后处理 OffscreenComponent,再处理 lazy 节点

然后就是正常的渲染流程了,流程结束 懒加载的组件 替换 fallback 展示到页面上

最后

以上就是我梳理的 suspense 和 lazy 组合的工作流程,感谢大家的阅读,有不对的地方也欢迎大家指出来。

参考

  • React 18.2.0 源码
相关推荐
小高0071 分钟前
📈前端图片压缩实战:体积直降 80%,LCP 提升 2 倍
前端·javascript·面试
OEC小胖胖5 分钟前
【React Hooks】封装的艺术:如何编写高质量的 React 自-定义 Hooks
前端·react.js·前端框架·web
BillKu13 分钟前
vue3+element-plus 输入框el-input设置背景颜色和字体颜色,样式效果等同于不可编辑的效果
前端·javascript·vue.js
惊悚的毛毛虫17 分钟前
掘金免广告?不想看理财交流圈?不想看exp+8?
前端
springfe010123 分钟前
vue3组件 - 大文件上传
前端·vue.js
再学一点就睡31 分钟前
Vite 工作原理(简易版)—— 从代码看核心逻辑
前端·vite
NeilNiu41 分钟前
开源AI工具Midscene.js
javascript·人工智能·开源
好好好明天会更好1 小时前
uniapp项目中小程序的生命周期
前端·vue.js
CF14年老兵1 小时前
「Vue 3 + View Transition 实现炫酷圆形缩放换肤动画」
前端·css·trae
Warren981 小时前
软件测试-Selenium学习笔记
java·javascript·笔记·学习·selenium·测试工具·安全