vue中的虛擬dom知識(shí)點(diǎn)總結(jié)
一、簡介
虛擬dom是隨著時(shí)代發(fā)展而誕生的產(chǎn)物。 命令式操作dom:
- 簡單易用
- 程序中的狀態(tài)很難管理,代碼中的邏輯也很混亂。
現(xiàn)在三大主流框架都是聲明式操作DOM。通過描述狀態(tài)和DOM之間的映射關(guān)系,就可以將狀態(tài)渲染成視圖。狀態(tài)可以是JS 中的任意類型。將狀態(tài)作為輸入,并生成DOM輸出到頁面上顯示出來的過程叫渲染。 通常程序在運(yùn)行時(shí),狀態(tài)會(huì)不斷發(fā)生變化。狀態(tài)發(fā)生變化時(shí),就需要重新渲染。 如何確定狀態(tài)中發(fā)生了什么變化以及需要在哪里更新DOM? 最簡單粗暴的解決方式:把所有DOM全刪了,使用狀態(tài)重新生成一份DOM,輸出到頁面上顯示出來。但是訪問dom是非常昂貴的,這樣會(huì)造成相當(dāng)多的性能浪費(fèi)。
虛擬DOM的解決方式:通過狀態(tài)生成虛擬節(jié)點(diǎn)樹,然后使用虛擬節(jié)點(diǎn)樹進(jìn)行渲染。在渲染之前,會(huì)對(duì)比新舊虛擬DOM,只渲染不同的部分。 虛擬DOM是將狀態(tài)映射為視圖的眾多解決方案中的一種。
二、為什么VUE引入了虛擬DOM?
vue在一定程度上知道具體哪個(gè)狀態(tài)發(fā)生了變化和哪些節(jié)點(diǎn)使用了這個(gè)狀態(tài),這樣就可以通過更細(xì)粒度的綁定來更新視圖。代價(jià)是粒度太細(xì)會(huì)有很多watcher同時(shí)觀察某些狀態(tài),會(huì)有一些內(nèi)存開銷以及一些依賴追蹤的開銷。vue2.0采取了一個(gè)中等粒度的解決方案,狀態(tài)偵測(cè)只細(xì)化到某個(gè)組件,組件內(nèi)部通過虛擬DOM渲染視圖,這可以大大縮減依賴數(shù)量和watcher數(shù)量。 vue通過模板描述狀態(tài)和視圖之間的映射關(guān)系,它會(huì)先將模板編譯成渲染函數(shù),然后執(zhí)行渲染函數(shù)生成虛擬節(jié)點(diǎn),使用虛擬節(jié)點(diǎn)更新視圖。
虛擬DOM在vue中的作用:
提供虛擬節(jié)點(diǎn)和對(duì)比新舊vnode,并根據(jù)對(duì)比結(jié)果進(jìn)行DOM操作來更新視圖。
三、vnode
VNode類可以實(shí)例化不同類型的vnode實(shí)例,而不同類型的vnode實(shí)例各自表示不同類型的DOM元素。vnode可以理解為節(jié)點(diǎn)描述對(duì)象,描述了應(yīng)該怎么創(chuàng)建真實(shí)DOM。
1.vnode的類型
- 注釋節(jié)點(diǎn):只有兩個(gè)有效屬性:isComment屬性和text屬性
- 文本節(jié)點(diǎn):只有text屬性
- 元素節(jié)點(diǎn):tag,data(節(jié)點(diǎn)的數(shù)據(jù):如attrs、class、style等),children,context(當(dāng)前組件的vue實(shí)例)
- 組件節(jié)點(diǎn):獨(dú)有屬性為componentOptions(組件節(jié)點(diǎn)的選項(xiàng)參數(shù)),componentInstance(組件實(shí)例)
- 函數(shù)式組件:類似于組件節(jié)點(diǎn),獨(dú)有屬性為functionalContext和functionalOptions
- 克隆節(jié)點(diǎn):將現(xiàn)有節(jié)點(diǎn)的屬性復(fù)制到新節(jié)點(diǎn)中,與被克隆節(jié)點(diǎn)的唯一區(qū)別是isCloned屬性,克隆節(jié)點(diǎn)的isCloned為true。作用是優(yōu)化靜態(tài)節(jié)點(diǎn)和插槽節(jié)點(diǎn)。
四、虛擬dom的核心——patch
可以將vnode渲染成真實(shí)的DOM,實(shí)際作用是計(jì)算出真正需要更新的節(jié)點(diǎn),在現(xiàn)有DOM上進(jìn)行修改來實(shí)現(xiàn)更新視圖的目的,最大限度減少DOM操作,顯著提升性能。本質(zhì)是用js的運(yùn)算成本替換DOM操作的執(zhí)行成本,js的運(yùn)算速度比DOM快得多,所以很劃算。
1.patch的過程
就是以新的vnode為準(zhǔn)增刪改節(jié)點(diǎn)。
新增節(jié)點(diǎn)
- oldVnode不存在而vnode中存在。通常發(fā)生在首次渲染中。
- vnode和oldVnode完全不是同一個(gè)節(jié)點(diǎn),vnode是新節(jié)點(diǎn),而oldVnode是被廢棄的節(jié)點(diǎn)。
刪除節(jié)點(diǎn)
- 節(jié)點(diǎn)只在oldVnode中存在。
替換過程是將新創(chuàng)建的DOM節(jié)點(diǎn)插入到舊節(jié)點(diǎn)的旁邊,然后再將舊節(jié)點(diǎn)刪除。
更新節(jié)點(diǎn)
增刪節(jié)點(diǎn)的共同點(diǎn)是兩個(gè)虛擬節(jié)點(diǎn)完全不同。新舊節(jié)點(diǎn)是同一個(gè)文本節(jié)點(diǎn),需要進(jìn)行更細(xì)致的比對(duì)。
靜態(tài)節(jié)點(diǎn)
靜態(tài)節(jié)點(diǎn):一旦渲染到頁面上就不會(huì)變化。 如果新舊兩個(gè)虛擬節(jié)點(diǎn)時(shí)靜態(tài)節(jié)點(diǎn)就直接跳過更新節(jié)點(diǎn)。
新虛擬節(jié)點(diǎn)有文本屬性
新虛擬節(jié)點(diǎn)有文本屬性而且和舊節(jié)點(diǎn)文本屬性不一樣,就直接調(diào)用setTextContent方法(在瀏覽器環(huán)境下是node.textContent方法)來將視圖中DOM節(jié)點(diǎn)的內(nèi)容改為虛擬節(jié)點(diǎn)的text屬性所保存的文本。
新虛擬節(jié)點(diǎn)無文本屬性
新虛擬節(jié)點(diǎn)有children:
- 舊虛擬節(jié)點(diǎn)也有children屬性,對(duì)新舊兩個(gè)虛擬節(jié)點(diǎn)的children進(jìn)行更詳細(xì)的對(duì)比并更新(增刪改移)。
- 舊虛擬節(jié)點(diǎn)沒有children屬性,說明舊虛擬節(jié)點(diǎn)要么是一個(gè)空標(biāo)簽,要么是有文本的文本節(jié)點(diǎn)。如果是文本節(jié)點(diǎn),先把文本清空讓它變成空標(biāo)簽,然后將新虛擬節(jié)點(diǎn)(vnode)中的children挨個(gè)創(chuàng)建成真實(shí)的DOM元素節(jié)點(diǎn)并將其插入到視圖中的DOM節(jié)點(diǎn)下面。
新虛擬節(jié)點(diǎn)無children:
沒有text也沒有children,說明是空節(jié)點(diǎn)。把舊虛擬節(jié)點(diǎn)刪成空標(biāo)簽。
五、更新子節(jié)點(diǎn)的優(yōu)化策略
雙端對(duì)比。 只需要嘗試使用相同位置的兩個(gè)節(jié)點(diǎn)來比對(duì)是否是同一個(gè)節(jié)點(diǎn):如果恰巧是同一個(gè)節(jié)點(diǎn),直接就可以進(jìn)入更新節(jié)點(diǎn)的操作;如果嘗試失敗了,再用循環(huán)的方式來查找節(jié)點(diǎn)。這樣做可以很大程度地避免循環(huán)oldChildren來查找節(jié)點(diǎn),從而使執(zhí)行速度得到很大的提升。
- oldChildren先循環(huán)完畢,如果newChildren中還有剩余的節(jié)點(diǎn),說明這些節(jié)點(diǎn)都是需要新增的節(jié)點(diǎn),直接把這些節(jié)點(diǎn)插入到DOM中就行了。
- newChildren先循環(huán)完畢,、如果oldChildren還有剩余的節(jié)點(diǎn),、說明oldChildren中剩余的節(jié)點(diǎn)都是被廢棄的節(jié)點(diǎn),是應(yīng)該被刪除的節(jié)點(diǎn)。不需要循環(huán)對(duì)比就可以知道需要將這些節(jié)點(diǎn)從DOM中移除。
到此這篇關(guān)于vue中的虛擬dom知識(shí)點(diǎn)總結(jié)的文章就介紹到這了,更多相關(guān)vue中的虛擬dom內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
vue項(xiàng)目中實(shí)現(xiàn)el-dialog組件可拖拽效果
本文主要介紹了vue項(xiàng)目中實(shí)現(xiàn)el-dialog組件可拖拽效果,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-01-01vue項(xiàng)目運(yùn)行npm?install報(bào)錯(cuò)問題及解決
這篇文章主要介紹了vue項(xiàng)目運(yùn)行npm?install報(bào)錯(cuò)問題及解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-08-08vue-cli3+typescript初體驗(yàn)小結(jié)
這篇文章主要介紹了vue-cli3+typescript初體驗(yàn)小結(jié),小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2019-02-02關(guān)于在vue 中使用百度ueEditor編輯器的方法實(shí)例代碼
這篇文章主要介紹了在vue 中使用百度ueEditor編輯器的方法實(shí)例代碼 ,非常不錯(cuò),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2018-09-09vue3.x源碼剖析之?dāng)?shù)據(jù)響應(yīng)式的深入講解
這篇文章主要給大家介紹了關(guān)于vue3.x源碼剖析之?dāng)?shù)據(jù)響應(yīng)式的相關(guān)資料,在講解過程中,我們會(huì)對(duì)比Vue2.x的API特性,使用有哪些區(qū)別,需要的朋友可以參考下2022-01-01vue后臺(tái)管理之動(dòng)態(tài)加載路由的方法
這篇文章主要介紹了vue后臺(tái)管理之動(dòng)態(tài)加載路由的方法,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2018-08-08