成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

Snabbdom.js(三)

dantezhao / 725人閱讀

摘要:總共寫了四篇文章都是自己的一些拙見,僅供參考,請多多指教,我這邊也會持續(xù)修正加更新介紹一下基本用法介紹一下渲染原理介紹一下的算法和對值的認識介紹一下對于兼容的修改這篇主要是說一下的算法在上一篇中我總結(jié)過對比渲染的流程大體分為通過來判斷兩個是

總共寫了四篇文章(都是自己的一些拙見,僅供參考,請多多指教,我這邊也會持續(xù)修正加更新)

介紹一下snabbdom基本用法

介紹一下snabbdom渲染原理

介紹一下snabddom的diff算法和對key值的認識

介紹一下對于兼容IE8的修改

這篇主要是說一下snabbdom的diff算法

在上一篇中我總結(jié)過:
對比渲染的流程大體分為
1.通過sameVnode來判斷兩個vnode是否值得進行比較
2.如果不值得,直接刪除舊的vnode,渲染新的vnode
3.如果值得,調(diào)用模塊鉤子函數(shù),對其節(jié)點的屬性進行替換,例如style,event等;再判斷節(jié)點子節(jié)點是否為文本節(jié)點,如果為文本節(jié)點則進行更替,如果還存在其他子節(jié)點則調(diào)用updateChildren,對子節(jié)點進行更新,更新流程將會回到第一步,重復(fù);

這篇文章的重點就是說一下updateChildren這個函數(shù)

sameVnode
function sameVnode(vnode1, vnode2) {
    return vnode1.key === vnode2.key && vnode1.sel === vnode2.sel;
}

這是一個比較兩個vnode是否相似,是否值得去進行比較的函數(shù),那么這里為什么會提到它?因為這里面有一個很重要的值---key
在平時的使用中幾乎用不到這個key值,不會去專門給它一個定義值,因為undefined===undefined,不會影響其比較;

key的作用

key值的出現(xiàn)主要是為了應(yīng)付一些場景
例如:

    • 1
    • 2
    • 2
    • -->
    • 3
    • 3
    • 4

對于這種情況,如果按照正常的做法,就是一個個vnode去進行比較,發(fā)現(xiàn)其文本節(jié)點不對,就會一個個進行替換例如:

  • 1
  • -->
  • 2
  • .... 這樣就會進行三次dom操作

    對于這種情況是否可以優(yōu)化呢?
    答案是可以的,我們可以刪除

  • 1
  • ,然后添加一個
  • 4
  • ,這樣就只進行了兩次dom操作就完成了需要的效果

    那這里就涉及到一個標記值,標記著在新vnode中還有哪些舊的vnode存在,key值就是充當著這個角色。

    [1(key:a),2(key:b),3(key:c)]

    [2(key:b),3(key:c),4(key:d)]

    [a,b,c] -> [a(x),b,c,d(+)] === [1,2,3] --> [1(x),2,3,4(+)]

    key值與vnode形成了一個映射,可以看到,我們通過對key值的排序、增刪間接完成了對vnode的操作,使用最少的dom操作來完成了

    如何對key值進行排序,增刪

    那這里就會有一個問題,我們?nèi)绾瓮瓿缮厦娴牟僮髂??這個過程我們可以理解為一種優(yōu)化對比渲染的過程,也就是diff算法的核心

    建議大家先看一下這一篇文章,圖文并茂

    我這邊舉一個復(fù)雜的例子,記錄每一步的操作:

    下面是頁面真實的dom,分別保存在自己vnode的elm屬性上;舊-->新

      • a
      • a
      • b
      • d
      • c
      • f
      • f
      • h
      • e
      • -->
      • k
      • d
      • b
      • g
      • g

    假設(shè)每個元素都有一個key值一一對應(yīng),且不重復(fù),它們的key值分別為

        a:1            a:1
        b:2            d:6
        c:3            f:4
        f:4    -->     h:8
        e:5            k:9
        d:6            b:2
        g:7            g:7

    將舊新vnode分別放入兩個數(shù)組

    old:[vnode,....] 
    new:[vnode,....]
    

    其實我們是比較其key值是否相等,然后再決定如何排序,增刪vnode的位置,patch vnode,最終達到改變dom的目的,為了方便理解,我這里把其key值拿出來放入一個數(shù)組,每一個key在數(shù)組中的索引都對應(yīng)著相應(yīng)的vnode在其數(shù)組中的索引,在真實代碼中是直接比較vnode.key值。

    oldKey:[1,2,3,4,5,6,7] 
    
    oldStartIdx:0
    oldStartVal:1
    
    oldEndIdx:6
    oldEndVal:7
    
    newKey:[1,6,4,8,9,2,7]
    
    newStartIdx:0
    newStartVal:1
    
    newEndIdx:6
    newEndVal:7
    

    用的是雙指針的方法,頭尾同時開始掃描;
    循環(huán)兩個數(shù)組,循環(huán)條件為(old_startIndex <= old_endIndex && new_startIndex <= new_endIndex)

    (下面說的patch是直接對vnoe.elm進行修改,調(diào)用前面的patchVnode函數(shù),也就是直接對頁面的dom進行修改,及時比較及時修改)

    比較oldStartVal和newStartVal是否相等,如果相等則oldStartIdx和newStartIdx分別加1,并對oldStartVal對應(yīng)的vnode進行patch,進入下一次循環(huán);這個例子中oldStartVal==newStartVal,所以oldStartIdx:1 newStartIdx:1;若不相等,繼續(xù)比較;

     比較過后:
       oldStartIdx:1              oldEndIdx:6
       oldStartVal:2              oldEndVal:7
    
    
       newStartIdx:1              newEndIdx:6     
       newStartVal:6              newEndVal:7
    
    
    比較范圍縮小后:
       
       oldKey:[2,3,4,5,6,7] 
       newKey:[6,4,8,9,2,7]
       
    dom:
       
  • a
  • b
  • c
  • f
  • e
  • d
  • g
  • oldVnodeArray:舊的vnode數(shù)組 [a,b,c,f,e,d,g]

    比較oldEndVal和newEndVal是否相等,如果相等則oldEndIdx和newEndIdx分別減1,并對oldEndVal對應(yīng)的舊vnode進行patch,進入下一次循環(huán);這里oldEndVal==newEndVal,所以oldEndIdx:5 newEndIdx:5;若不相等,繼續(xù)比較;

    比較過后:
       oldStartIdx:1              oldEndIdx:5
       oldStartVal:2              oldEndVal:6
    
    
       newStartIdx:1              newEndIdx:5     
       newStartVal:6              newEndVal:2
    
    比較范圍縮小后:
       
       oldKey:[2,3,4,5,6] 
       newKey:[6,4,8,9,2]
    
    dom:
       
  • a
  • b
  • c
  • f
  • e
  • d
  • g
  • oldVnodeArray:舊的vnode數(shù)組 [a,b,c,f,e,d,g]

    比較oldStartVal和newEndVal是否相等,如果相等則oldStartIdx和newEndIdx分別加1和減1,oldStartVal對應(yīng)的vnode移動到oldEndVal對應(yīng)的vnode后面,并對移動的vnode進行patch,進入下一次循環(huán);這里oldStartVal==newEndVal,所以oldStartIdx:2 newEndIdx:4;若不相等,繼續(xù)比較;

    比較過后:
       oldStartIdx:2              oldEndIdx:5
       oldStartVal:3              oldEndVal:6
    
    
       newStartIdx:1              newEndIdx:4     
       newStartVal:6              newEndVal:9
    
    比較范圍縮小后:
       
       oldKey:[3,4,5,6] 
       newKey:[6,4,8,9]
    dom:
       
  • a
  • c
  • f
  • e
  • d
  • b
  • g
  • oldVnodeArray:舊的vnode數(shù)組 [a,b,c,f,e,d,g]

    比較oldEndVal和newStartVal是否相等,如果相等則oldEndIdx和newStartIdx分別減1和加1,oldEndVal對應(yīng)的vnode移動到oldStart對應(yīng)的vnode前面,并對移動的vnode進行patch,進入下一次循環(huán);這里oldEndVal==newStartVal,所以oldEndIdx:4 newStartIdx:2;若不相等,繼續(xù)比較;

    比較過后:
       oldStartIdx:2              oldEndIdx:4
       oldStartVal:3              oldEndVal:5
    
    
       newStartIdx:2              newEndIdx:4     
       newStartVal:4              newEndVal:9
    
    比較范圍縮小后:
       
       oldKey:[3,4,5] 
       newKey:[4,8,9]
       
    dom:
       
  • a
  • d
  • c
  • f
  • e
  • b
  • g
  • oldVnodeArray:舊的vnode數(shù)組 [a,b,c,f,e,d,g]

    若不滿足上述判斷條件,查找newStartVal對應(yīng)的vnode是否存在于舊vnode數(shù)組中。若存在,移動這個舊的vnode到oldStartVal對應(yīng)的vnode前面,并對這個移動的vnode進行patch,在舊的vnode數(shù)組中將其原來的位置置為undefined,并且newStartIdx加1;

    比較過后:
       oldStartIdx:2              oldEndIdx:4
       oldStartVal:3              oldEndVal:5
    
    
       newStartIdx:3              newEndIdx:4     
       newStartVal:8              newEndVal:9
       
    
    比較范圍縮小后:
       
       oldKey:[3,4,5] 
       newKey:[8,9]
       
    dom:
       
  • a
  • d
  • f
  • c
  • e
  • b
  • g
  • oldVnodeArray:舊的vnode數(shù)組 [a,b,c,undefined,e,d,g]

    若不存在,則將這個newStartVal對應(yīng)的vnde添加到oldStartVal對應(yīng)的vnode前面,并且newStartIdx加1;

     比較過后:
       oldStartIdx:2              oldEndIdx:4
       oldStartVal:3              oldEndVal:5
    
    
       newStartIdx:4              newEndIdx:4     
       newStartVal:9              newEndVal:9
    
    比較范圍縮小后:
       
       oldKey:[3,4,5] 
       newKey:[9]
       
    dom:
       
  • a
  • d
  • f
  • h
  • c
  • e
  • b
  • g
  • oldVnodeArray:舊的vnode數(shù)組 [a,b,c,undefined,e,d,g] 這里循環(huán)了兩次 比較過后: oldStartIdx:2 oldEndIdx:4 oldStartVal:3 oldEndVal:5 newStartIdx:5 newEndIdx:4 newStartVal:undefined newEndVal:9 比較范圍縮小后: oldKey:[3,4,5] newKey:[] dom:
  • a
  • d
  • f
  • h
  • k
  • c
  • e
  • b
  • g
  • oldVnodeArray:舊的vnode數(shù)組 [a,b,c,undefined,e,d,g]

    循環(huán)結(jié)束,判斷新舊vnode的key值哪個遍歷完,如果舊的便利完,若舊vnode數(shù)組遍歷完,則將剩余的新vnode數(shù)組中的vnode進行添加;若新vnode數(shù)組遍歷完,則刪除剩余的舊vnode數(shù)組中的vnode
    在上面例子中,我們需要刪除oldVnodeArray中的三個vnode,索引分別為3,4,5,從而刪除了vnode對應(yīng)的elm
    最后得到最終的dom結(jié)構(gòu)

               
    • a
    • d
    • f
    • h
    • k
    • b
    • g

    上面的例子沒有將所有情況全部歸納進來,不過應(yīng)該包含了大部分情況了。還需要注意的就是:

    上面只是提到了key值,其實比較兩個vnode是否相似還有一個sel屬性,必須要兩個都相等才行

    正常情況下key值用到的地方也是ul-li tr-td這種子元素重復(fù)的場景,因為這種情況下才會涉及到子元素順序改變還能復(fù)用

    通過上面的分析,其實還可以發(fā)現(xiàn)一個key值的特點,就是唯一性和一一對應(yīng)性。唯一性好理解,畢竟key值就是用來每個vnode自己的標示;一一對應(yīng)代表著是你舊vnode和新vnode中如果沒有改變,則其key值應(yīng)保持不變,之所以要提這個是因為很多地方看到了進行循環(huán)渲染的時候其key值都是用的數(shù)組的index進行賦值

    如果考慮這種情況
    
  • a
  • c
  • b
  • -->
  • a
  • c
  • b
  • 一般這種dom結(jié)構(gòu)都是放在數(shù)組里面循環(huán)輸出的,如果它們的key值是按照index進行賦值的話,就需要這個地方需要進行三次dom操作,就是依次修改其節(jié)點的文本值; [a(1),b(2),c(3)] [c(1),a(2),b(3)] 那如果我使得它們改變前面對應(yīng)的元素的key值不改變的,一一對應(yīng)的話,這里只需要一次dom操作,就是把
  • c
  • 移動到最前面 [a(1),b(2),c(3)] [c(3),a(1),b(2)]

    文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

    轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/100371.html

    相關(guān)文章

    • React && VUE Virtual Dom的Diff算法統(tǒng)一之路 snabbd

      摘要:毫無疑問的是算法的復(fù)雜度與效率是決定能夠帶來性能提升效果的關(guān)鍵因素。速度略有損失,但可讀性大大提高。因此目前的主流算法趨向一致,在主要思路上,與的方式基本相同。在里面實現(xiàn)了的算法與支持。是唯一添加的方法所以只發(fā)生在中。 VirtualDOM是react在組件化開發(fā)場景下,針對DOM重排重繪性能瓶頸作出的重要優(yōu)化方案,而他最具價值的核心功能是如何識別并保存新舊節(jié)點數(shù)據(jù)結(jié)構(gòu)之間差異的方法,...

      shixinzhang 評論0 收藏0
    • Snabbdom.js(一)

      摘要:閑聊在學(xué)的過程中,虛擬應(yīng)該是聽的最多的概念之一,得知其是借鑒進行開發(fā),故習(xí)之。以我的觀點來看,多個相同元素渲染時,則需要為每個元素添加值。 閑聊:在學(xué)vue的過程中,虛擬dom應(yīng)該是聽的最多的概念之一,得知其是借鑒snabbdom.js進行開發(fā),故習(xí)之。由于我工作處于IE8的環(huán)境,對ES6,TS這些知識的練習(xí)也只是淺嘗輒止,而snabbdom.js從v.0.5.4這個版本后開始使用TS...

      mating 評論0 收藏0
    • snabbdom源碼粗讀

      摘要:這個大概是的鉤子吧在每一次插入操作的時候都將節(jié)點這類型方法可以看出來是在調(diào)用對應(yīng)的方法因為開始的時候就導(dǎo)入進來了插入節(jié)點操作的時候都需要加入子節(jié)點有子元素也就是的時候遞歸調(diào)用循環(huán)子節(jié)點生成對應(yīng)著一些操作之后都要觸發(fā)鉤子函數(shù)。 snabbdom 本文的snabbdom源碼分析采用的是0.54版本(即未用ts重寫前的最后一版) 前期了解 snabbdom被用作vue的虛擬dom。本文的一個...

      svtter 評論0 收藏0
    • snabbdom.js(四)

      摘要:總共寫了四篇文章都是自己的一些拙見,僅供參考,請多多指教,我這邊也會持續(xù)修正加更新介紹一下基本用法介紹一下渲染原理介紹一下的算法和對值的認識介紹一下對于兼容的修改這篇主要是記錄一下針對做了哪些修改增加用來兼容某些功能函數(shù),例如等將每個文件單 總共寫了四篇文章(都是自己的一些拙見,僅供參考,請多多指教,我這邊也會持續(xù)修正加更新) 介紹一下snabbdom基本用法 介紹一下snabbdo...

      wuyangchun 評論0 收藏0
    • snabbdom.js(二)

      摘要:如果新舊的和都相同,說明兩個相似,我們就可以保留舊的節(jié)點,再具體去比較其差異性,在舊的上進行打補丁否則直接替換節(jié)點。 總共寫了四篇文章(都是自己的一些拙見,僅供參考,請多多指教,我這邊也會持續(xù)修正加更新) 介紹一下snabbdom基本用法 介紹一下snabbdom渲染原理 介紹一下snabddom的diff算法和對key值的認識 介紹一下對于兼容IE8的修改 這篇我將以自己的思路去...

      浠ラ箍 評論0 收藏0

    發(fā)表評論

    0條評論

    閱讀需要支付1元查看
    <