react 源码解析 6.legacy 模式和 concurrent 模式
react 源码解析 6.legacy 模式和 concurrent 模式
视频讲解(高效学习):进入学习
往期文章:
react 启动的模式
react 有 3 种模式进入主体函数的入口,我们可以从 react 官方文档 <a href="https://zh-hans.reactjs.org/docs/concurrent-mode-adoption.html#feature-comparison">使用 Concurrent 模式(实验性)</a>中对比三种模式:
legacy 模式:
ReactDOM.render(<App />, rootNode)
。这是当前 React app 使用的方式。当前没有计划删除本模式,但是这个模式可能不支持这些新功能。blocking 模式:
ReactDOM.createBlockingRoot(rootNode).render(<App />)
。目前正在实验中。作为迁移到 concurrent 模式的第一个步骤。concurrent 模式:
ReactDOM.createRoot(rootNode).render(<App />)
。目前在实验中,未来稳定之后,打算作为 React 的默认开发模式。这个模式开启了所有的新功能。
特性对比:
legacy 模式在合成事件中有自动批处理的功能,但仅限于一个浏览器任务。非 React 事件想使用这个功能必须使用 unstable_batchedUpdates
。在 blocking 模式和 concurrent 模式下,所有的 setState
在默认情况下都是批处理的。会在开发中发出警告
不同模式在 react 运行时的含义
legacy 模式是我们常用的,它构建 dom 的过程是同步的,所以在 render 的 reconciler 中,如果 diff 的过程特别耗时,那么导致的结果就是 js 一直阻塞高优先级的任务(例如用户的点击事件),表现为页面的卡顿,无法响应。
concurrent Mode 是 react 未来的模式,它用时间片调度实现了异步可中断的任务,根据设备性能的不同,时间片的长度也不一样,在每个时间片中,如果任务到了过期时间,就会主动让出线程给高优先级的任务。这部分将在第 15 节 scheduler&lane 模型 。
两种模式函数主要执行过程
1.主要执行流程:
2.详细函数调用过程:
用 demo_0 跟着视频调试更加清晰,黄色部分是主要任务是创建 fiberRootNode 和 rootFiber,红色部分是创建 Update,蓝色部分是调度 render 阶段的入口函数
3.legacy 模式:
render 调用 legacyRenderSubtreeIntoContainer,最后 createRootImpl 会调用到 createFiberRoot 创建 fiberRootNode,然后调用 createHostRootFiber 创建 rootFiber,其中 fiberRootNode 是整个项目的的根节点,rootFiber 是当前应用挂在的节点,也就是 ReactDOM.render 调用后的根节点
创建完 Fiber 节点后,legacyRenderSubtreeIntoContainer 调用 updateContainer 创建创建 Update 对象挂载到 updateQueue 的环形链表上,然后执行 scheduleUpdateOnFiber 调用 performSyncWorkOnRoot 进入 render 阶段和 commit 阶段
4.concurrent 模式:
createRoot 调用 createRootImpl 创建 fiberRootNode 和 rootNode
创建完 Fiber 节点后,调用 ReactDOMRoot.prototype.render 执行 updateContainer,然后 scheduleUpdateOnFiber 异步调度 performConcurrentWorkOnRoot 进入 render 阶段和 commit 阶段
5.legacy 模式主要函数注释
6.concurrent 主要函数注释:
7.两种模式的不同点:
createRootImpl 中传入的第二个参数不一样 一个是 LegacyRoot 一个是 ConcurrentRoot
requestUpdateLane 中获取的 lane 的优先级不同
在函数 scheduleUpdateOnFiber 中根据不同优先级进入不同分支,legacy 模式进入 performSyncWorkOnRoot,concurrent 模式会异步调度 performConcurrentWorkOnRoot
评论