網(wǎng)站首頁 編程語言 正文
VDOM(虛擬dom)
react和vue都是基于vdom的前端框架。
web界面由DOM樹來構建,當其中一部分發(fā)生變化時,其實就是對應的某個節(jié)點發(fā)生了變化。
若一次操作中有十次更新DOM的動作,虛擬DOM不會立即操作DOM,而是將這十次更新的diff內容保存到本地的一個js對象中,最終將這個js對象一次性attach到DOM樹上,再進行后續(xù)的操作。
用js對象模擬DOM節(jié)點的好處是:
- 頁面的更新可以先全部反映在js對象(虛擬DOM)上,精準的對比關心的屬性,避免大量無謂的計算。等更新完成后,再將最終的js對象映射成真是的DOM,交由瀏覽器去繪制。
- 為應用帶來的跨平臺的能力,不再僅僅局限于瀏覽器端。比如:React-Native、canvas等。
//虛擬dom對象
{
type: 'div',
props: {
id: 'aaa',
className: ['bbb', 'ccc'],
onClick: function() {}
},
children: []
}
react中,jsx由babel轉義再經(jīng)過render后生成我們想要的VDOM(虛擬DOM)。
Fiber架構
react15的時候,和 vue 的渲染流程還是很像的,都是遞歸渲染 vdom,增刪改 dom 就行。
但是因為狀態(tài)管理的差異導致了架構的差異。react的setState會渲染整個vdom,而一個應用的所有 vdom 可能是很龐大的,計算量就可能很大。
瀏覽器里 js 計算時間太長是會阻塞渲染的,會占用每一幀的動畫、重繪重排的時間,這樣動畫就會卡頓。
那能不能把計算量拆分一下,每一幀計算一部分,不要阻塞動畫的渲染呢?
順著這個思路,react 就改造為了 fiber 架構,目標是打斷計算,分多次進行。
渲染的時候不要直接更新到 dom 了,只找到變化的部分,打個增刪改的標記,創(chuàng)建好 dom,等全部計算完了一次性更新到 dom 就好了。
初始化渲染
根據(jù) React Element 生成對應的 Fiber 樹
1.根據(jù) React Element 生成對應的 Fiber 樹
首次執(zhí)行ReactDOM.render 會創(chuàng)建fiberRoot 和 rootFiber。
- fiberRoot:整個應用的根節(jié)點(只能有一個)
- rootFiber:組件樹的根節(jié)點(可以有多個)
2.根據(jù)組件返回的JSX在內存中依次創(chuàng)建Fiber節(jié)點,并連接在一起構建形成Fiber樹,被稱為workInProgress Fiber樹。最后以 workInProgress 作為最新的渲染樹,fiberRoot 的 current 指針指向 workInProgress 使其變?yōu)?current Fiber 樹。到此完成初始化流程。
- workInProgress:正在內存中構建的 Fiber 樹稱為 workInProgress Fiber樹。在一次更新中,所有的更新都是發(fā)生在 workInProgress 樹上。在一次更新之后,workInProgress 樹上的狀態(tài)是最新的狀態(tài),那么它將變成 current 樹用于渲染視圖。
- current:正在視圖層渲染的樹叫做 current 樹。
更新時
render階段,創(chuàng)建 dom,打上增刪改的 tag,等全部計算完之后,commit 階段一次性更新到 dom。
render階段
beginWork和completeWork階段會循環(huán)最新的jsx生成的虛擬dom,通過對比虛擬dom和current Fiber樹生成workinProgress Fiber樹。
beginWork
創(chuàng)建本次循環(huán)主體的子Fiber節(jié)點
- mount(首屏渲染)時創(chuàng)建子Fiber節(jié)點,并返回改新建節(jié)點;
- update時若不滿足復用條件,則與mount時一樣創(chuàng)建新的子fiber節(jié)點,并diff出相應的effTag掛在FIber節(jié)點上,并返回該新建節(jié)點;
- updata時若滿足復用條件,且判斷仍需處理其子節(jié)點的后代,則返回復用后的子Fiber節(jié)點
- updata時若滿足復用條件,且判斷不需繼續(xù)處理其子節(jié)點的后代,則直接返回null值;
completeWork
構建或更新DOM節(jié)點
- 構建過程中,會自下而上將子節(jié)點插入到當前節(jié)點
- 更新過程中,會計算DOM 節(jié)點的屬性,一旦屬性需要更新,會為DOM節(jié)點對應的workINProgress節(jié)點標記Update的effectTag
自下而上收集effectList,最終收集到root上
Diff到底是誰跟誰比?
Diff 是 vdom 和 current Fiber 對比,生成 workInProgressFiber
effectTag與effectList
render階段不會真正操作dom,只會創(chuàng)建dom然后打個effectTag的增刪改標記。commit階段就根據(jù)標記來更新dom就可以了。
但是commit階段要在遍歷一次fiber來查找有effectTag的節(jié)點,更新dom嗎?
當然沒問題,但是顯然很低效。完全可以把有 effectTag 的節(jié)點收集到一個鏈表里,然后 commit 階段直接遍歷這個鏈表就行了。這個鏈表叫做 effectList。
react 會在 commit 階段遍歷 effectList,根據(jù) effectTag 來增刪改 dom。
這個隊列叫做 effectList。
commit階段
render階段找到變化的部分,創(chuàng)建dom,打上增刪改的tag,等全部計算完之后,commit階段一次性更新到dom。
before mutation階段(執(zhí)行DOM操作前)
遍歷effectList,依次執(zhí)行:
1.處理DOM節(jié)點渲染/刪除后的autoFocus、blur邏輯
2.調用getSnapshotBeforeUpdate生命周期鉤子
3.調度useEffect mutation階段(執(zhí)行DOM操作)
遍歷 effectList 來更新 dom
layout階段(執(zhí)行DOM操作后)
因為這個階段已經(jīng)可以拿到布局信息了,會同步調用 useLayoutEffect 的回調函數(shù)。而且這個階段可以拿到新的 dom 節(jié)點,還會更新下 ref。
原文鏈接:https://blog.csdn.net/weixin_44247866/article/details/125950721
相關推薦
- 2022-07-17 C#實現(xiàn)子類與父類的相互轉換_C#教程
- 2022-07-11 docker搭建redis 主從哨兵集群
- 2022-09-08 Python實現(xiàn)聚類K-means算法詳解_python
- 2022-07-26 面向對象OOP基礎理解
- 2022-11-03 tomcat?集群監(jiān)控與彈性伸縮詳解_Tomcat
- 2023-03-28 Linux版本中Nginx平滑升級與回退_nginx
- 2023-07-08 qt修改默認構建路徑
- 2022-05-29 ASP.NET?Core使用HttpClient調用WebService_實用技巧
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細win安裝深度學習環(huán)境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結構-簡單動態(tài)字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支