摘要:的過程就是調(diào)用函數(shù),就像打補丁一樣修改真實。兩個節(jié)點值得比較時,會調(diào)用函數(shù)這是很重要的一步,讓引用到現(xiàn)在的真實,當(dāng)修改時,會同步變化。兩個節(jié)點都有子節(jié)點,而且它們不一樣,這樣我們會調(diào)用函數(shù)比較子節(jié)點,這是的核心,后邊會講到。
轉(zhuǎn)載請注明出處
本文轉(zhuǎn)載至我的blog
目錄前言
virtual dom
分析diff
總結(jié)
前言vue2.0加入了virtual dom,有向react靠攏的意思。vue的diff位于patch.js文件中,我的一個小框架aoy也同樣使用此算法,該算法來源于snabbdom,復(fù)雜度為O(n)。
了解diff過程可以讓我們更高效的使用框架。
本文力求以圖文并茂的方式來講明這個diff的過程。
如果不了解virtual dom,要理解diff的過程是比較困難的。虛擬dom對應(yīng)的是真實dom, 使用document.CreateElement 和 document.CreateTextNode創(chuàng)建的就是真實節(jié)點。
我們可以做個試驗。打印出一個空元素的第一層屬性,可以看到標(biāo)準(zhǔn)讓元素實現(xiàn)的東西太多了。如果每次都重新生成新的元素,對性能是巨大的浪費。
var mydiv = document.createElement("div"); for(var k in mydiv ){ console.log(k) }
virtual dom就是解決這個問題的一個思路,到底什么是virtual dom呢?通俗易懂的來說就是用一個簡單的對象去代替復(fù)雜的dom對象。
舉個簡單的例子,我們在body里插入一個class為a的div。
var mydiv = document.createElement("div"); mydiv.className = "a"; document.body.appendChild(mydiv);
對于這個div我們可以用一個簡單的對象mydivVirtual代表它,它存儲了對應(yīng)dom的一些重要參數(shù),在改變dom之前,會先比較相應(yīng)虛擬dom的數(shù)據(jù),如果需要改變,才會將改變應(yīng)用到真實dom上。
//偽代碼 var mydivVirtual = { tagName: "DIV", className: "a" }; var newmydivVirtual = { tagName: "DIV", className: "b" } if(mydivVirtual.tagName !== newmydivVirtual.tagName || mydivVirtual.className !== newmydivVirtual.className){ change(mydiv) } // 會執(zhí)行相應(yīng)的修改 mydiv.className = "b"; //最后讀到這里就會產(chǎn)生一個疑問,為什么不直接修改dom而需要加一層virtual dom呢?
很多時候手工優(yōu)化dom確實會比virtual dom效率高,對于比較簡單的dom結(jié)構(gòu)用手工優(yōu)化沒有問題,但當(dāng)頁面結(jié)構(gòu)很龐大,結(jié)構(gòu)很復(fù)雜時,手工優(yōu)化會花去大量時間,而且可維護(hù)性也不高,不能保證每個人都有手工優(yōu)化的能力。至此,virtual dom的解決方案應(yīng)運而生,virtual dom很多時候都不是最優(yōu)的操作,但它具有普適性,在效率、可維護(hù)性之間達(dá)平衡。
virtual dom 另一個重大意義就是提供一個中間層,js去寫ui,ios安卓之類的負(fù)責(zé)渲染,就像reactNative一樣。
分析diff一篇相當(dāng)經(jīng)典的文章React’s diff algorithm中的圖,react的diff其實和vue的diff大同小異。所以這張圖能很好的解釋過程。比較只會在同層級進(jìn)行, 不會跨層級比較。
舉個形象的例子。
aoy diff
aoy
diff
我們可能期望將直接移動到
的后邊,這是最優(yōu)的操作。但是實際的diff操作是移除
里的在創(chuàng)建一個新的插到
的后邊。
因為新加的在層級2,舊的在層級3,屬于不同層級的比較。
文中的代碼位于aoy-diff中,已經(jīng)精簡了很多代碼,留下最核心的部分。
diff的過程就是調(diào)用patch函數(shù),就像打補丁一樣修改真實dom。
function patch (oldVnode, vnode) { if (sameVnode(oldVnode, vnode)) { patchVnode(oldVnode, vnode) } else { const oEl = oldVnode.el let parentEle = api.parentNode(oEl) createEle(vnode) if (parentEle !== null) { api.insertBefore(parentEle, vnode.el, api.nextSibling(oEl)) api.removeChild(parentEle, oldVnode.el) oldVnode = null } } return vnode }
patch函數(shù)有兩個參數(shù),vnode和oldVnode,也就是新舊兩個虛擬節(jié)點。在這之前,我們先了解完整的vnode都有什么屬性,舉個一個簡單的例子:
// body下的對應(yīng)的 oldVnode 就是 { el: div //對真實的節(jié)點的引用,本例中就是document.querySelector("#id.classA") tagName: "DIV", //節(jié)點的標(biāo)簽 sel: "div#v.classA" //節(jié)點的選擇器 data: null, // 一個存儲節(jié)點屬性的對象,對應(yīng)節(jié)點的el[prop]屬性,例如onclick , style children: [], //存儲子節(jié)點的數(shù)組,每個子節(jié)點也是vnode結(jié)構(gòu) text: null, //如果是文本節(jié)點,對應(yīng)文本節(jié)點的textContent,否則為null }需要注意的是,el屬性引用的是此 virtual dom對應(yīng)的真實dom,patch的vnode參數(shù)的el最初是null,因為patch之前它還沒有對應(yīng)的真實dom。
來到patch的第一部分,
if (sameVnode(oldVnode, vnode)) { patchVnode(oldVnode, vnode) }sameVnode函數(shù)就是看這兩個節(jié)點是否值得比較,代碼相當(dāng)簡單:
function sameVnode(oldVnode, vnode){ return vnode.key === oldVnode.key && vnode.sel === oldVnode.sel }兩個vnode的key和sel相同才去比較它們,比如p和span,div.classA和div.classB都被認(rèn)為是不同結(jié)構(gòu)而不去比較它們。
如果值得比較會執(zhí)行patchVnode(oldVnode, vnode),稍后會詳細(xì)講patchVnode函數(shù)。
當(dāng)節(jié)點不值得比較,進(jìn)入else中
else { const oEl = oldVnode.el let parentEle = api.parentNode(oEl) createEle(vnode) if (parentEle !== null) { api.insertBefore(parentEle, vnode.el, api.nextSibling(oEl)) api.removeChild(parentEle, oldVnode.el) oldVnode = null } }過程如下:
取得oldvnode.el的父節(jié)點,parentEle是真實dom
createEle(vnode)會為vnode創(chuàng)建它的真實dom,令vnode.el =真實dom
parentEle將新的dom插入,移除舊的dom
當(dāng)不值得比較時,新節(jié)點直接把老節(jié)點整個替換了最后
return vnodepatch最后會返回vnode,vnode和進(jìn)入patch之前的不同在哪?
沒錯,就是vnode.el,唯一的改變就是之前vnode.el = null, 而現(xiàn)在它引用的是對應(yīng)的真實dom。var oldVnode = patch (oldVnode, vnode)至此完成一個patch過程。
patchVnode兩個節(jié)點值得比較時,會調(diào)用patchVnode函數(shù)
patchVnode (oldVnode, vnode) { const el = vnode.el = oldVnode.el let i, oldCh = oldVnode.children, ch = vnode.children if (oldVnode === vnode) return if (oldVnode.text !== null && vnode.text !== null && oldVnode.text !== vnode.text) { api.setTextContent(el, vnode.text) }else { updateEle(el, vnode, oldVnode) if (oldCh && ch && oldCh !== ch) { updateChildren(el, oldCh, ch) }else if (ch){ createEle(vnode) //create el"s children dom }else if (oldCh){ api.removeChildren(el) } } }const el = vnode.el = oldVnode.el 這是很重要的一步,讓vnode.el引用到現(xiàn)在的真實dom,當(dāng)el修改時,vnode.el會同步變化。
節(jié)點的比較有5種情況
if (oldVnode === vnode),他們的引用一致,可以認(rèn)為沒有變化。
if(oldVnode.text !== null && vnode.text !== null && oldVnode.text !== vnode.text),文本節(jié)點的比較,需要修改,則會調(diào)用Node.textContent = vnode.text。
if( oldCh && ch && oldCh !== ch ), 兩個節(jié)點都有子節(jié)點,而且它們不一樣,這樣我們會調(diào)用updateChildren函數(shù)比較子節(jié)點,這是diff的核心,后邊會講到。
else if (ch),只有新的節(jié)點有子節(jié)點,調(diào)用createEle(vnode),vnode.el已經(jīng)引用了老的dom節(jié)點,createEle函數(shù)會在老dom節(jié)點上添加子節(jié)點。
else if (oldCh),新節(jié)點沒有子節(jié)點,老節(jié)點有子節(jié)點,直接刪除老節(jié)點。
updateChildrenupdateChildren (parentElm, oldCh, newCh) { let oldStartIdx = 0, newStartIdx = 0 let oldEndIdx = oldCh.length - 1 let oldStartVnode = oldCh[0] let oldEndVnode = oldCh[oldEndIdx] let newEndIdx = newCh.length - 1 let newStartVnode = newCh[0] let newEndVnode = newCh[newEndIdx] let oldKeyToIdx let idxInOld let elmToMove let before while (oldStartIdx <= oldEndIdx && newStartIdx <= newEndIdx) { if (oldStartVnode == null) { //對于vnode.key的比較,會把oldVnode = null oldStartVnode = oldCh[++oldStartIdx] }else if (oldEndVnode == null) { oldEndVnode = oldCh[--oldEndIdx] }else if (newStartVnode == null) { newStartVnode = newCh[++newStartIdx] }else if (newEndVnode == null) { newEndVnode = newCh[--newEndIdx] }else if (sameVnode(oldStartVnode, newStartVnode)) { patchVnode(oldStartVnode, newStartVnode) oldStartVnode = oldCh[++oldStartIdx] newStartVnode = newCh[++newStartIdx] }else if (sameVnode(oldEndVnode, newEndVnode)) { patchVnode(oldEndVnode, newEndVnode) oldEndVnode = oldCh[--oldEndIdx] newEndVnode = newCh[--newEndIdx] }else if (sameVnode(oldStartVnode, newEndVnode)) { patchVnode(oldStartVnode, newEndVnode) api.insertBefore(parentElm, oldStartVnode.el, api.nextSibling(oldEndVnode.el)) oldStartVnode = oldCh[++oldStartIdx] newEndVnode = newCh[--newEndIdx] }else if (sameVnode(oldEndVnode, newStartVnode)) { patchVnode(oldEndVnode, newStartVnode) api.insertBefore(parentElm, oldEndVnode.el, oldStartVnode.el) oldEndVnode = oldCh[--oldEndIdx] newStartVnode = newCh[++newStartIdx] }else { // 使用key時的比較 if (oldKeyToIdx === undefined) { oldKeyToIdx = createKeyToOldIdx(oldCh, oldStartIdx, oldEndIdx) // 有key生成index表 } idxInOld = oldKeyToIdx[newStartVnode.key] if (!idxInOld) { api.insertBefore(parentElm, createEle(newStartVnode).el, oldStartVnode.el) newStartVnode = newCh[++newStartIdx] } else { elmToMove = oldCh[idxInOld] if (elmToMove.sel !== newStartVnode.sel) { api.insertBefore(parentElm, createEle(newStartVnode).el, oldStartVnode.el) }else { patchVnode(elmToMove, newStartVnode) oldCh[idxInOld] = null api.insertBefore(parentElm, elmToMove.el, oldStartVnode.el) } newStartVnode = newCh[++newStartIdx] } } } if (oldStartIdx > oldEndIdx) { before = newCh[newEndIdx + 1] == null ? null : newCh[newEndIdx + 1].el addVnodes(parentElm, before, newCh, newStartIdx, newEndIdx) }else if (newStartIdx > newEndIdx) { removeVnodes(parentElm, oldCh, oldStartIdx, oldEndIdx) } }代碼很密集,為了形象的描述這個過程,可以看看這張圖。
過程可以概括為:oldCh和newCh各有兩個頭尾的變量StartIdx和EndIdx,它們的2個變量相互比較,一共有4種比較方式。如果4種比較都沒匹配,如果設(shè)置了key,就會用key進(jìn)行比較,在比較的過程中,變量會往中間靠,一旦StartIdx>EndIdx表明oldCh和newCh至少有一個已經(jīng)遍歷完了,就會結(jié)束比較。
具體的diff分析設(shè)置key和不設(shè)置key的區(qū)別:
不設(shè)key,newCh和oldCh只會進(jìn)行頭尾兩端的相互比較,設(shè)key后,除了頭尾兩端的比較外,還會從用key生成的對象oldKeyToIdx中查找匹配的節(jié)點,所以為節(jié)點設(shè)置key可以更高效的利用dom。diff的遍歷過程中,只要是對dom進(jìn)行的操作都調(diào)用api.insertBefore,api.insertBefore只是原生insertBefore的簡單封裝。
比較分為兩種,一種是有vnode.key的,一種是沒有的。但這兩種比較對真實dom的操作是一致的。對于與sameVnode(oldStartVnode, newStartVnode)和sameVnode(oldEndVnode,newEndVnode)為true的情況,不需要對dom進(jìn)行移動。
總結(jié)遍歷過程,有3種dom操作:
當(dāng)oldStartVnode,newEndVnode值得比較,說明oldStartVnode.el跑到oldEndVnode.el的后邊了。
圖中假設(shè)startIdx遍歷到1。
當(dāng)oldEndVnode,newStartVnode值得比較,oldEndVnode.el跑到了oldStartVnode.el的前邊,準(zhǔn)確的說應(yīng)該是oldEndVnode.el需要移動到oldStartVnode.el的前邊”。
newCh中的節(jié)點oldCh里沒有, 將新節(jié)點插入到oldStartVnode.el的前邊。
在結(jié)束時,分為兩種情況:
oldStartIdx > oldEndIdx,可以認(rèn)為oldCh先遍歷完。當(dāng)然也有可能newCh此時也正好完成了遍歷,統(tǒng)一都?xì)w為此類。此時newStartIdx和newEndIdx之間的vnode是新增的,調(diào)用addVnodes,把他們?nèi)坎暹M(jìn)before的后邊,before很多時候是為null的。addVnodes調(diào)用的是insertBefore操作dom節(jié)點,我們看看insertBefore的文檔:parentElement.insertBefore(newElement, referenceElement)
如果referenceElement為null則newElement將被插入到子節(jié)點的末尾。如果newElement已經(jīng)在DOM樹中,newElement首先會從DOM樹中移除。所以before為null,newElement將被插入到子節(jié)點的末尾。newStartIdx > newEndIdx,可以認(rèn)為newCh先遍歷完。此時oldStartIdx和oldEndIdx之間的vnode在新的子節(jié)點里已經(jīng)不存在了,調(diào)用removeVnodes將它們從dom里刪除。
下面舉個例子,畫出diff完整的過程,每一步dom的變化都用不同顏色的線標(biāo)出。a,b,c,d,e假設(shè)是4個不同的元素,我們沒有設(shè)置key時,b沒有復(fù)用,而是直接創(chuàng)建新的,刪除舊的。
當(dāng)我們給4個元素加上唯一key時,b得到了的復(fù)用。
這個例子如果我們使用手工優(yōu)化,只需要3步就可以達(dá)到。
總結(jié)盡量不要跨層級的修改dom
設(shè)置key可以最大化的利用節(jié)點
不要盲目相信diff的效率,在必要時可以手工優(yōu)化
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://specialneedsforspecialkids.com/yun/82125.html
相關(guān)文章
前端知識庫 - 收藏集 - 掘金
摘要:自適應(yīng)的橢圓背景知識屬性的基本用法使用樣式畫各種圖形前端掘金下面是一些我在中經(jīng)常用到的圖案,還有一些是在看到的。像圖手把手教你使用前端掘金使用教程一是什么是目前世界上最先進(jìn)的分布式版本控制系統(tǒng)。 如何在 Vue.js 中使用第三方庫 - 前端 - 掘金在諸多 Vue.js 應(yīng)用中, Lodash, Moment, Axios, Async等都是一些非常有用的 JavaScript 庫....
前端知識庫 - 收藏集 - 掘金
摘要:自適應(yīng)的橢圓背景知識屬性的基本用法使用樣式畫各種圖形前端掘金下面是一些我在中經(jīng)常用到的圖案,還有一些是在看到的。像圖手把手教你使用前端掘金使用教程一是什么是目前世界上最先進(jìn)的分布式版本控制系統(tǒng)。 如何在 Vue.js 中使用第三方庫 - 前端 - 掘金在諸多 Vue.js 應(yīng)用中, Lodash, Moment, Axios, Async等都是一些非常有用的 JavaScript 庫....
virtualDom的DIFF算法關(guān)鍵過程整理
摘要:,文本節(jié)點的比較,需要修改,則會調(diào)用。兩個節(jié)點都有子節(jié)點,而且它們不一樣,這樣我們會調(diào)用函數(shù)比較子節(jié)點,這是的核心。,新節(jié)點沒有子節(jié)點,老節(jié)點有子節(jié)點,直接刪除老節(jié)點。參考文章解析的算法 判斷對應(yīng)節(jié)點是否有必要進(jìn)行比較(sameVnode) function sameVnode(oldVnode, vnode){ return vnode.key === oldVnode.ke...
javascript高級學(xué)習(xí)總結(jié)(二)
摘要:那個率先改變的實例的返回值,就會傳遞給的回調(diào)函數(shù)。函數(shù)對函數(shù)的改進(jìn),體現(xiàn)在以下四點內(nèi)置執(zhí)行器。進(jìn)一步說,函數(shù)完全可以看作多個異步操作,包裝成的一個對象,而命令就是內(nèi)部命令的語法糖。中的本質(zhì)就是沒有的隱藏的組件。 1、原型 - jquery使用showImg(https://segmentfault.com/img/bVbwNcY?w=692&h=442);注釋 : 實例雖然不同,但是構(gòu)...
發(fā)表評論
0條評論
閱讀 3070·2023-04-25 16:50
閱讀 904·2021-11-25 09:43
閱讀 3512·2021-09-26 10:11
閱讀 2518·2019-08-26 13:28
閱讀 2531·2019-08-26 13:23
閱讀 2419·2019-08-26 11:53
閱讀 3566·2019-08-23 18:19
閱讀 2987·2019-08-23 16:27