Web應(yīng)用在不斷地發(fā)展,但是用戶對(duì)網(wǎng)站速度和響應(yīng)時(shí)間的要求也在不斷提高。如何降低網(wǎng)站的響應(yīng)時(shí)間,提升用戶體驗(yàn)已經(jīng)成為Web開(kāi)發(fā)者的重要任務(wù)。Vue.js是一款輕量級(jí)MVVM框架,通過(guò)Virtual DOM技術(shù),Vue.js實(shí)現(xiàn)了數(shù)據(jù)的雙向綁定,使得開(kāi)發(fā)者在開(kāi)發(fā)Web應(yīng)用時(shí)更加高效且方便。其中Vue的dom diff算法是Vue的核心之一。
在Vue.js中,頁(yè)面的更新是基于Virtual DOM技術(shù),而Virtual DOM本質(zhì)上是一個(gè)JavaScript對(duì)象,它通過(guò)一系列操作(比如,增刪改查等)來(lái)描述真實(shí)的DOM樹(shù)。在更新頁(yè)面時(shí),我們需要對(duì)比新舊兩個(gè)虛擬DOM之間的差異,并僅對(duì)需要更新的部分進(jìn)行更新。
function diff(oldTree, newTree) { // todo... }
diff算法的思路是先通過(guò)遞歸方式深度遍歷新舊虛擬DOM的節(jié)點(diǎn),然后記錄下節(jié)點(diǎn)的差異,并嘗試復(fù)用舊節(jié)點(diǎn)和舊節(jié)點(diǎn)中的子節(jié)點(diǎn)。這個(gè)過(guò)程中涉及到DOM樹(shù)的建立、屬性的設(shè)置、文本的替換等操作。通過(guò)dom diff的思想,讓我們節(jié)省了大量的更新成本和計(jì)算時(shí)間。
一個(gè)節(jié)點(diǎn)的比較過(guò)程通常是由以下3個(gè)步驟組成:
- 節(jié)點(diǎn)類型變化:節(jié)點(diǎn)從原來(lái)的文本節(jié)點(diǎn)變成了復(fù)雜節(jié)點(diǎn)
- 屬性變化:例如class, style等發(fā)生了變化
- 文本變化
function updateProp(oldVal, newVal) { // 如果新舊值不一樣 if(oldVal !== newVal) { // 如果新屬性值是null或undefined if(newVal === null || newVal === undefined) { node.removeAttribute('key') } // 如果新的屬性值是對(duì)象 else if(typeof newVal === 'object') { node.setAttribute('key', JSON.stringify(newVal)) } // 基本類型直接設(shè)置即可 else { node.setAttribute('key', newVal) } } }
改變節(jié)點(diǎn)的內(nèi)容通常有兩種方式:一種是用節(jié)點(diǎn)的innerHTML屬性,另一種是通過(guò)子節(jié)點(diǎn)的分別更新。第一種執(zhí)行簡(jiǎn)單快速,但是,它會(huì)將整個(gè)節(jié)點(diǎn)的子樹(shù)移除,并重新插入,有可能造成一些不必要的浪費(fèi)。
Vue的dom diff算法能夠減少大量的重繪和回流計(jì)算,使得頁(yè)面的響應(yīng)更加迅速,增強(qiáng)了用戶的體驗(yàn)。Vue雖然是一款相對(duì)比較輕量化的框架,但它提供了足夠的功能和Api,可以方便地完成大多數(shù)業(yè)務(wù)需求。