专栏名称: SegmentFault思否
SegmentFault (www.sf.gg)开发者社区,是中国年轻开发者喜爱的极客社区,我们为开发者提供最纯粹的技术交流和分享平台。
目录
相关文章推荐
程序员小灰  ·  部署DeepSeek ... ·  12 小时前  
程序员小灰  ·  看完《哪吒2》,热泪盈眶! ·  昨天  
码农翻身  ·  漫画 | 为什么大家都愿意进入外企? ·  2 天前  
程序员的那些事  ·  OpenAI ... ·  3 天前  
程序员的那些事  ·  印度把 DeepSeek ... ·  4 天前  
51好读  ›  专栏  ›  SegmentFault思否

解析 vue2.0 的 diff 算法

SegmentFault思否  · 公众号  · 程序员  · 2017-09-15 08:00

正文

本文转载至我的blog:

https://github.com/aooy/blog

前言

vue2.0加入了virtual dom,有向react靠拢的意思。vue的diff位于patch.js文件中,我的一个小框架aoy也同样使用此算法,该算法来源于snabbdom,复杂度为O(n)。了解diff过程可以让我们更高效的使用框架。

本文力求以图文并茂的方式来讲明这个diff的过程。

virtual dom

如果不了解virtual dom,要理解diff的过程是比较困难的。虚拟dom对应的是真实dom, 使用 document.CreateElement document.CreateTextNode 创建的就是真实节点。

我们可以做个试验。打印出一个空元素的第一层属性,可以看到标准让元素实现的东西太多了。如果每次都重新生成新的元素,对性能是巨大的浪费。

  1. var mydiv = document.createElement('div');

  2. for(var k in mydiv ){

  3.  console.log(k)

  4. }

virtual dom就是解决这个问题的一个思路,到底什么是virtual dom呢?通俗易懂的来说就是用一个简单的对象去代替复杂的dom对象。

举个简单的例子,我们在body里插入一个class为a的div。

  1. var mydiv = document.createElement('div');

  2. mydiv.className = 'a';

  3. document.body.appendChild(mydiv);

对于这个div我们可以用一个简单的对象 mydivVirtual 代表它,它存储了对应dom的一些重要参数,在改变dom之前,会先比较相应虚拟dom的数据,如果需要改变,才会将改变应用到真实dom上。

  1. //伪代码

  2. var mydivVirtual = {

  3.  tagName: 'DIV',

  4.  className: 'a'

  5. };

  6. var newmydivVirtual = {

  7.   tagName: 'DIV',

  8.   className: 'b'

  9. }

  10. if(mydivVirtual.tagName !== newmydivVirtual.tagName || mydivVirtual.className  !== newmydivVirtual.className){

  11.   change(mydiv)

  12. }

  13. // 会执行相应的修改 mydiv.className = 'b';

  14. //最后  

读到这里就会产生一个疑问,为什么不直接修改dom而需要加一层virtual dom呢?

很多时候手工优化dom确实会比virtual dom效率高,对于比较简单的dom结构用手工优化没有问题,但当页面结构很庞大,结构很复杂时,手工优化会花去大量时间,而且可维护性也不高,不能保证每个人都有手工优化的能力。至此,virtual dom的解决方案应运而生, virtual dom很多时候都不是最优的操作,但它具有普适性,在效率、可维护性之间达平衡。

virtual dom 另一个重大意义就是提供一个中间层,js去写ui,ios安卓之类的负责渲染,就像reactNative一样。

分析diff

一篇相当经典的文章React’s diff algorithm中的图,react的diff其实和vue的diff大同小异。所以这张图能很好的解释过程。比较只会在同层级进行, 不会跨层级比较。

举个形象的例子。

  1.          
  2.  

               

  3.     aoy    

  4.    diffSpan>

  5.  

  • aoy

  • diff Span >

  • 我们可能期望将 直接移动到

    的后边,这是最优的操作。但是实际的diff操作是移除

    里的 在创建一个新的 插到

    的后边。 因为新加的 在层级2,旧的在层级3,属于不同层级的比较。

    源码分析

    文中的代码位于aoy-diff中,已经精简了很多代码,留下最核心的部分。

    diff的过程就是调用patch函数,就像打补丁一样修改真实dom。

    1. function patch (oldVnode, vnode) {

    2.    if (sameVnode(oldVnode, vnode)) {

    3.        patchVnode(oldVnode, vnode)

    4.    } else {

    5.        const oEl = oldVnode.el

    6.        let parentEle = api.parentNode(oEl)

    7.        createEle(vnode)

    8.         if (parentEle !== null) {

    9.            api.insertBefore(parentEle, vnode.el, api.nextSibling(oEl))

    10.            api.removeChild(parentEle, oldVnode.el)

    11.            oldVnode = null

    12.        }

    13.    }

    14.    return vnode

    15. }

    patch 函数有两个参数, vnode oldVnode ,也就是新旧两个虚拟节点。在这之前,我们先了解完整的vnode都有什么属性,举个一个简单的例子:

    1. // body下的

      对应的 oldVnode 就是
    2. {

    3.  el:  div   //对真实的节点的引用,本例中就是document.querySelector('#id.classA')

    4.  tagName: 'DIV',   //节点的标签

    5.  sel: 'div#v.classA'  //节点的选择器

    6.  data: null,       // 一个存储节点属性的对象,对应节点的el[prop]属性,例如onclick , style

    7.  children: [], //存储子节点的数组,每个子节点也是vnode结构

    8.  text: null,    //如果是文本节点,对应文本节点的textContent,否则为null

    9. }

    10. 需要注意的是,el属性引用的是此 virtual dom对应的真实dom, patchvnode参数的 el最初是null,因为 patch之前它还没有对应的真实dom。

      来到 patch 的第一部分,

      1. if (sameVnode(oldVnode, vnode)) {

      2.    patchVnode(oldVnode, vnode)

      3. }

      sameVnode函数就是看这两个节点是否值得比较,代码相当简单:

      1. function sameVnode(oldVnode, vnode){

      2.    return vnode.key === oldVnode.key && vnode.sel === oldVnode.sel

      3. }

      两个vnode的key和sel相同才去比较它们,比如 pspandiv.classAdiv.classB 都被认为是不同结构而不去比较它们。

      如果值得比较会执行 patchVnode(oldVnode,vnode),稍后会详细讲 patchVnode函数。

      当节点不值得比较,进入else中

      1. else {

      2.        const oEl = oldVnode.el

      3.        let parentEle = api.parentNode(oEl)

      4.        createEle(vnode)

      5.        if (parentEle !== null) {

      6.            api.insertBefore(parentEle, vnode.el, api.nextSibling(oEl))

      7.            api.removeChild(parentEle, oldVnode.el)

      8.            oldVnode = null

      9.        }

      10.    }

      过程如下:

      • 取得 oldvnode.el的父节点, parentEle是真实dom

      • createEle(vnode)会为 vnode创建它的真实dom,令 vnode.el = 真实dom

      • parentEle将新的dom插入,移除旧的dom 当不值得比较时,新节点直接把老节点整个替换了

      最后

      1. return vnode

      patch最后会返回vnode,vnode和进入patch之前的不同在哪?

      没错,就是vnode.el,唯一的改变就是之前vnode.el = null, 而现在它引用的是对应的真实dom。

      1. var oldVnode = patch (oldVnode, vnode)

      至此完成一个patch过程。

      patchVnode

      两个节点值得比较时,会调用 patchVnode函数

      1. patchVnode (oldVnode, vnode) {

      2.    const el = vnode.el = oldVnode.el

      3.    let i, oldCh = oldVnode.children, ch = vnode.children

      4.    if (oldVnode === vnode) return

      5.     if (oldVnode.text !== null && vnode.text !== null && oldVnode.text !== vnode.text) {

      6.        api.setTextContent(el, vnode.text)

      7.    }else {

      8.        updateEle(el, vnode, oldVnode)

      9.        if (oldCh && ch && oldCh !== ch) {

      10.            updateChildren(el, oldCh, ch)

      11.        }else if (ch){

      12.            createEle(vnode) //create el's children dom

      13.        } else if (oldCh){

      14.            api.removeChildren(el)

      15.        }

      16.    }

      17. }

      constel=vnode.el=oldVnode.el 这是很重要的一步,让 vnode.el引用到现在的真实dom,当 el修改时, vnode.el会同步变化。

      节点的比较有5种情况:

      1. if(oldVnode===vnode),他们的引用一致,可以认为没有变化。

      2. if(oldVnode.text!==null&&vnode.text!==null&&oldVnode.text!==vnode.text),文本节点的比较,需要修改,则会调用 Node.textContent=vnode.text

      3. if(oldCh&&ch&&oldCh!==ch) , 两个节点都有子节点,而且它们不一样,这样我们会调用 updateChildren函数比较子节点,这是diff的核心,后边会讲到。

      4. elseif(ch),只有新的节点有子节点,调用 createEle(vnode), vnode.el已经引用了老的dom节点, createEle函数会在老dom节点上添加子节点。

      5. elseif(oldCh),新节点没有子节点,老节点有子节点,直接删除老节点。

      updateChildren

      1. updateChildren (parentElm, oldCh, newCh) {

      2.    let oldStartIdx = 0, newStartIdx = 0

      3.    let oldEndIdx = oldCh.length - 1

      4.    let oldStartVnode = oldCh[0]

      5.    let oldEndVnode = oldCh[oldEndIdx]

      6.    let newEndIdx = newCh.length - 1

      7.    let newStartVnode = newCh[0]

      8.    let newEndVnode = newCh[newEndIdx]

      9.    let oldKeyToIdx

      10.    let idxInOld

      11.    let elmToMove

      12.    let before

      13.    while (oldStartIdx <= oldEndIdx && newStartIdx <= newEndIdx) {

      14.            if (oldStartVnode == null) {   //对于vnode.key的比较,会把oldVnode = null

      15.                oldStartVnode = oldCh[++oldStartIdx]

      16.            }else if (oldEndVnode == null) {

      17.                oldEndVnode = oldCh[--oldEndIdx]

      18.            }else if (newStartVnode == null) {

      19.                newStartVnode = newCh[++newStartIdx]

      20.            }else if (newEndVnode == null) {

      21.                newEndVnode = newCh[--newEndIdx]

      22.            }else if (sameVnode(oldStartVnode, newStartVnode)) {

      23.                patchVnode(oldStartVnode, newStartVnode)

      24.                oldStartVnode = oldCh[++oldStartIdx]

      25.                newStartVnode = newCh[++newStartIdx]

      26.            }else if (sameVnode(oldEndVnode, newEndVnode)) {

      27.                patchVnode(oldEndVnode, newEndVnode)

      28.                oldEndVnode = oldCh[--oldEndIdx]

      29.                newEndVnode = newCh[--newEndIdx]

      30.            }else if (sameVnode(oldStartVnode, newEndVnode)) {

      31.                patchVnode(oldStartVnode, newEndVnode)

      32.                api.insertBefore(parentElm, oldStartVnode.el, api.nextSibling(oldEndVnode.el))

      33.                oldStartVnode = oldCh[++oldStartIdx]

      34.                newEndVnode = newCh[--newEndIdx]

      35.            }else if (sameVnode(oldEndVnode, newStartVnode)) {

      36.                patchVnode(oldEndVnode, newStartVnode)

      37.                api.insertBefore(parentElm, oldEndVnode.el, oldStartVnode.el)

      38.                oldEndVnode = oldCh[--oldEndIdx]

      39.                newStartVnode = newCh[++newStartIdx]

      40.            }else {

      41.               // 使用key时的比较

      42.                if (oldKeyToIdx === undefined) {

      43.                    oldKeyToIdx = createKeyToOldIdx(oldCh, oldStartIdx, oldEndIdx) // 有key生成index表

      44.                }

      45.                idxInOld = oldKeyToIdx[newStartVnode.key]

      46.                if (!idxInOld) {







    请到「今天看啥」查看全文