如何實現一個 Virtual DOM 及源碼分析

如何實現一個 Virtual DOM 及源碼分析javascript

Virtual DOM算法html

    web頁面有一個對應的DOM樹,在傳統開發頁面時,每次頁面須要被更新時,都須要手動操做DOM來進行更新,可是咱們知道DOM操做對性能來講是很是不友好的,會影響頁面的重排,從而影響頁面的性能。所以在React和VUE2.0+引入了虛擬DOM的概念,他們的原理是:把真實的DOM樹轉換成javascript對象樹,也就是虛擬DOM,每次數據須要被更新的時候,它會生成一個新的虛擬DOM,而且和上次生成的虛擬DOM進行對比,對發生變化的數據作批量更新。---(由於操做JS對象會更快,更簡單,比操做DOM來講)。
咱們知道web頁面是由一個個HTML元素嵌套組合而成的,當咱們使用javascript來描述這些元素的時候,這些元素能夠簡單的被表示成純粹的JSON對象。java

好比以下HTML代碼:node

<div id="container" class="container">
   <ul id="list">
     <li class="item">111</li>
     <li class="item">222</li>
     <li class="item">333</li>
   </ul>
   <button class="btn btn-blue"><em>提交</em></button>
</div>

上面是真實的DOM樹結構,咱們可使用javascript中的json對象來表示的話,變成以下:git

var element = {
      tagName: 'div',
      props: {   // DOM的屬性
        id: 'container',
        class: 'container'
      },
      children: [
        {
          tagName: 'ul',
          props: {
            id: 'list'
          },
          children: [
            {tagName: 'li', props: {class: 'item'}, children: ['111']},
            {tagName: 'li', props: {class: 'item'}, children: ['222']},
            {tagName: 'li', props: {class: 'item'}, children: ['333']}
          ]
        },
        {
          tagName: 'button',
          props: {
            class: 'btn btn-blue'
          },
          children: [
            {
              tagName: 'em',
              children: ['提交']
            }
          ]
        }
      ]
   };

所以咱們可使用javascript對象表示DOM的信息和結構,當狀態變動的時候,從新渲染這個javascript對象的結構,而後可使用新渲染的對象樹去和舊的樹去對比,記錄兩顆樹的差別,兩顆樹的差別就是咱們須要對頁面真正的DOM操做,而後把他們應用到真正的DOM樹上,頁面就獲得更新。視圖的整個結構確實全渲染了,可是最後操做DOM的時候,只變動不一樣的地方。
所以咱們能夠總結一下 Virtual DOM算法:
1. 用javascript對象結構來表示DOM樹的結構,而後用這個樹構建一個真正的DOM樹,插入到文檔中。
2. 當狀態變動的時候,從新構造一顆新的對象樹,而後使用新的對象樹與舊的對象樹進行對比,記錄兩顆樹的差別。
3. 把記錄下來的差別用到步驟1所構建的真正的DOM樹上。視圖就更新了。github

算法實現:
2-1 使用javascript對象模擬DOM樹。
使用javascript來表示一個DOM節點,有如上JSON的數據,咱們只須要記錄它的節點類型,屬性和子節點便可。web

element.js 代碼以下:算法

function Element(tagName, props, children) {
  this.tagName = tagName;
  this.props = props;
  this.children = children;
}
Element.prototype.render = function() {
  var el = document.createElement(this.tagName);
  var props = this.props;
  // 遍歷子節點,依次設置子節點的屬性
  for (var propName in props) {
    var propValue = props[propName];
    el.setAttribute(propName, propValue);
  }
  // 保存子節點
  var childrens = this.children || [];
  // 遍歷子節點,使用遞歸的方式 渲染
  childrens.forEach(function(child) {
    var childEl = (child instanceof Element) ? child.render() // 若是子節點也是虛擬DOM,遞歸構建DOM節點
      : document.createTextNode(child);    // 若是是字符串的話,只構建文本節點
    el.appendChild(childEl);
  });
  return el;
};
module.exports = function(tagName, props, children) {
  return new Element(tagName, props, children);
}

入口index.js代碼以下:json

var el = require('./element');

var element = el('div', {id: 'container', class: 'container'}, [
  el('ul', {id: 'list'},[
    el('li', {class: 'item'}, ['111']),
    el('li', {class: 'item'}, ['222']),
    el('li', {class: 'item'}, ['333']),
  ]),
  el('button', {class: 'btn btn-blue'}, [
    el('em', {class: ''}, ['提交'])
  ])
]);

var elemRoot = element.render();
document.body.appendChild(elemRoot);

打開頁面便可看到效果。數組

2-2 比較兩顆虛擬DOM樹的差別及差別的地方進行dom操做

上面的div只會和同一層級的div對比,第二層級的只會和第二層級的對比,這樣的算法的複雜度能夠達到O(n).
可是在實際代碼中,會對新舊兩顆樹進行一個深度優先的遍歷,所以每一個節點都會有一個標記。以下圖所示:

在遍歷的過程當中,每次遍歷到一個節點就把該節點和新的樹進行對比,若是有差別的話就記錄到一個對象裏面。

如今咱們來看下個人目錄下 有哪些文件;而後分別對每一個文件代碼進行解讀,看看作了哪些事情,舊的虛擬dom和新的虛擬dom是如何比較的,且是如何更新頁面的 以下目錄:
目錄結構以下:

vdom  ---- 工程名
|   | ---- index.html  html頁面
|   | ---- element.js  實例化元素組成json數據 且 提供render方法 渲染頁面
|   | ---- util.js     提供一些公用的方法
|   | ---- diff.js     比較新舊節點數據 若是有差別保存到一個對象裏面去
|   | ---- patch.js    對當前差別的節點數據 進行DOM操做
|   | ---- index.js    頁面代碼初始化調用

首先是 index.js文件 頁面渲染完成後 變成以下html結構 

<div id="container">
  <h1 style="color: red;">simple virtal dom</h1>
  <p>the count is :1</p>
  <ul>
    <li>Item #0</li>
  </ul>
</div>

假如發生改變後,變成以下結構 

<div id="container">
  <h1 style="color: blue;">simple virtal dom</h1>
  <p>the count is :2</p>
  <ul>
    <li>Item #0</li>
    <li>Item #1</li>
  </ul>
</div>

能夠看到 新舊節點頁面數據的改變,h1標籤從屬性 顏色從紅色 變爲藍色,p標籤的文本發生改變,ul新增了一項元素li。
基本的原理是:先渲染出頁面數據出來,生成第一個模板頁面,而後使用定時器會生成一個新的頁面數據出來,對新舊兩顆樹進行一個深度優先的遍歷,所以每一個節點都會有一個標記。
而後調用diff方法對比對象新舊節點遍歷進行對比,找出二者的不一樣的地方存入到一個對象裏面去,最後經過patch.js找出對象不一樣的地方,分別進行dom操做。

index.js代碼以下:

var el = require('./element');
var diff = require('./diff');
var patch = require('./patch');

var count = 0;
function renderTree() {
  count++;
  var items = [];
  var color = (count % 2 === 0) ? 'blue' : 'red';
  for (var i = 0; i < count; i++) {
    items.push(el('li', ['Item #' + i]));
  }
  return el('div', {'id': 'container'}, [
    el('h1', {style: 'color: ' + color}, ['simple virtal dom']),
    el('p', ['the count is :' + count]),
    el('ul', items)
  ]);
}

var tree = renderTree()
var root = tree.render()
document.body.appendChild(root)
setInterval(function () {
  var newTree = renderTree()
  var patches = diff(tree, newTree)
  console.log(patches)
  patch(root, patches)
  tree = newTree
}, 1000);

執行 var tree = renderTree()方法後,會調用element.js,
1. 依次遍歷子節點(從內到外調用)依次爲 li, h1, p, ul, li和h1和p有一個文本子節點,所以遍歷完成後,count就等於1,
可是遍歷ul的時候,由於有一個子節點li,所以 count += 1; 因此調用完成後,ul的count等於2. 所以會對每一個element屬性添加count屬性。對於最外層的container容器就是對每一個子節點的依次增長,h1子節點默認爲1,循環完成後 +1;所以變爲2, p節點默認爲1,循環完成後 +1,所以也變爲2,ul爲2,循環完成後 +1,所以變爲3,所以container節點的count=2+2+3 = 7;

element.js部分代碼以下:

function Element(tagName, props, children) {
  if (!(this instanceof Element)) {
    // 判斷子節點 children 是否爲 undefined
    if (!utils.isArray(children) && children !== null) {
      children = utils.slice(arguments, 2).filter(utils.truthy);
    }
    return new Element(tagName, props, children);
  }
  // 若是沒有屬性的話,第二個參數是一個數組,說明第二個參數傳的是子節點
  if (utils.isArray(props)) {
    children = props;
    props = {};
  }
  this.tagName = tagName;
  this.props = props || {};
  this.children = children || [];
  // 保存key鍵 若是有屬性 保存key,不然返回undefined
  this.key = props ? props.key : void 0;
  var count = 0;
  
  utils.each(this.children, function(child, i) {
    // 若是是元素的實列的話
    if (child instanceof Element) {
      count += child.count;
    } else {
      // 若是是文本節點的話,直接賦值
      children[i] = '' + child;
    }
    count++;
  });
  this.count = count;
}

oldTree數據最終變成以下:

var oldTree = {
  tagName: 'div',
  key: undefined,
  count: 7,
  props: {id: 'container'},
  children: [
    {
      tagName: 'h1',
      key: undefined
      count: 1
      props: {style: 'colod: red'},
      children: ['simple virtal dom']
    },
    {
      tagName: 'p',
      key: undefined
      count: 1
      props: {},
      children: ['the count is :1']
    },
    {
      tagName: 'ul',
      key: undefined
      count: 2
      props: {},
      children: [
        {
          tagName: 'li',
          key: undefined,
          count: 1,
          props: {},
          children: ['Item #0']
        }
      ]
    },
  ]
};

定時器 執行 var newTree = renderTree()後,調用方法步驟仍是和第一步同樣:
2. 依次遍歷子節點(從內到外調用)依次爲 li, h1, p, ul, li和h1和p有一個文本子節點,所以遍歷完成後,count就等於1,由於有2個子元素li,count都爲1,所以ul每次遍歷依次在原來的基礎上加1,所以遍歷完成第一個li時候,ul中的count爲2,當遍歷完成第二個li的時候,ul的count就爲4了。所以ul中的count爲4. 對於最外層的container容器就是對每一個子元素依次增長。
因此 container節點的count = 2 + 2 + 5 = 9;

newTree數據最終變成以下數據:

var newTree = {
  tagName: 'div',
  key: undefined,
  count: 9,
  props: {id: 'container'},
  children: [
    {
      tagName: 'h1',
      key: undefined
      count: 1
      props: {style: 'colod: red'},
      children: ['simple virtal dom']
    },
    {
      tagName: 'p',
      key: undefined
      count: 1
      props: {},
      children: ['the count is :1']
    },
    {
      tagName: 'ul',
      key: undefined
      count: 4
      props: {},
      children: [
        {
          tagName: 'li',
          key: undefined,
          count: 1,
          props: {},
          children: ['Item #0']
        },
        {
          tagName: 'li',
          key: undefined,
          count: 1,
          props: {},
          children: ['Item #1']
        }
      ]
    },
  ]
}

var patches = diff(oldTree, newTree);

調用diff方法能夠比較新舊兩棵樹節點的數據,把兩顆樹的不一樣節點找出來。(注意,查看diff對比數據的方法,找到不一樣的節點,能夠查看這篇文章diff算法)以下調用代碼:

function diff (oldTree, newTree) {
  var index = 0;
  var patches = {};
  deepWalk(oldTree, newTree, index, patches);
  return patches;
}

執行deepWalk以下代碼:

function deepWalk(oldNode, newNode, index, patches) {
  var currentPatch = [];
  // 節點被刪除掉
  if (newNode === null) {
    // 真正的DOM節點時,將刪除執行從新排序,因此不須要作任何事
  } else if(utils.isString(oldNode) && utils.isString(newNode)) {
    // 替換文本節點
    if (newNode !== oldNode) {
      currentPatch.push({type: patch.TEXT, content: newNode});
    }
  } else if(oldNode.tagName === newNode.tagName && oldNode.key === newNode.key) {
    // 相同的節點,可是新舊節點的屬性不一樣的狀況下 比較屬性
    // diff props
    var propsPatches = diffProps(oldNode, newNode);
    if (propsPatches) {
      currentPatch.push({type: patch.PROPS, props: propsPatches});
    }
    // 不一樣的子節點 
    if (!isIgnoreChildren(newNode)) {
      diffChildren(
        oldNode.children,
        newNode.children,
        index,
        patches,
        currentPatch
      )
    }
  } else {
    // 不一樣的節點,那麼新節點替換舊節點
    currentPatch.push({type: patch.REPLACE, node: newNode});
  }
  if (currentPatch.length) {
    patches[index] = currentPatch;
  }
}

1. 判斷新節點是否爲null,若是爲null,說明節點被刪除掉。
2. 判斷新舊節點是否爲字符串,若是爲字符串說明是文本節點,而且新舊兩個文本節點不一樣的話,存入數組裏面去,以下代碼:

   currentPatch.push({type: patch.TEXT, content: newNode});
   patch.TEXT 爲 patch.js裏面的 TEXT = 3;content屬性爲新節點。

3. 若是新舊tagName相同的話,而且新舊節點的key相同的話,繼續比較新舊節點的屬性,以下代碼:

var propsPatches = diffProps(oldNode, newNode);

diffProps方法的代碼以下:

function diffProps(oldNode, newNode) {
      var count = 0;
      var oldProps = oldNode.props;
      var newProps = newNode.props;
      var key,
        value;
      var propsPatches = {};
      // 找出不一樣的屬性值
      for (key in oldProps) {
        value = oldProps[key];
        if (newProps[key] !== value) {
          count++;
          propsPatches[key] = newProps[key];
        }
      }
      // 找出新增屬性
      for (key in newProps) {
        value = newProps[key];
        if (!oldProps.hasOwnProperty(key)) {
          count++;
          propsPatches[key] = newProps[key];
        }
      }
      // 若是全部的屬性都是相同的話
      if (count === 0) {
        return null;
      }
      return propsPatches;
   }

diffProps代碼解析以下:

for (key in oldProps) {
   value = oldProps[key];
   if (newProps[key] !== value) {
      count++;
      propsPatches[key] = newProps[key];
   }
}

如上代碼是 判斷舊節點的屬性值是否在新節點中找到,若是找不到的話,count++; 把新節點的屬性值賦值給 propsPatches 存儲起來。

for (key in newProps) {
   value = newProps[key];
   if (!oldProps.hasOwnProperty(key)) {
      count++;
      propsPatches[key] = newProps[key];
   }
}

如上代碼是 判斷新節點的屬性是否能在舊節點中找到,若是找不到的話,count++; 把新節點的屬性值賦值給 propsPatches 存儲起來。

if (count === 0) {
   return null;
}
return propsPatches;

最後若是count 等於0的話,說明全部屬性都是相同的話,因此不須要作任何變化。不然的話,返回新增的屬性。

若是有 propsPatches 的話,執行以下代碼:

if (propsPatches) {
   currentPatch.push({type: patch.PROPS, props: propsPatches});
}

所以currentPatch數組裏面也有對應的更新的屬性,props就是須要更新的屬性對象。

繼續代碼:

// 不一樣的子節點 
if (!isIgnoreChildren(newNode)) {
   diffChildren(
     oldNode.children,
     newNode.children,
     index,
     patches,
     currentPatch
   )
}
function isIgnoreChildren(node) {
  return (node.props && node.props.hasOwnProperty('ignore'));
}

如上代碼判斷子節點是否相同,diffChildren代碼以下:

function diffChildren(oldChildren, newChildren, index, patches, currentPatch) {
    var diffs = listDiff(oldChildren, newChildren, 'key');
    newChildren = diffs.children;

    if (diffs.moves.length) {
      var recorderPatch = {type: patch.REORDER, moves: diffs.moves};
      currentPatch.push(recorderPatch);
    }

    var leftNode = null;
    var currentNodeIndex = index;
    utils.each(oldChildren, function(child, i) {
      var newChild = newChildren[i];
      currentNodeIndex = (leftNode && leftNode.count) ? currentNodeIndex + leftNode.count + 1 : currentNodeIndex + 1;
      // 遞歸
      deepWalk(child, newChild, currentNodeIndex, patches);
      leftNode = child;
    });
  }

如上代碼:var diffs = listDiff(oldChildren, newChildren, 'key'); 新舊節點按照key來比較,目前key爲undefined,因此diffs 爲以下:

diffs = {
    moves: [],
    children: [
      {
        tagName: 'h1',
        key: undefined
        count: 1
        props: {style: 'colod: blue'},
        children: ['simple virtal dom']
      },
      {
        tagName: 'p',
        key: undefined
        count: 1
        props: {},
        children: ['the count is :2']
      },
      {
        tagName: 'ul',
        key: undefined
        count: 4
        props: {},
        children: [
          {
            tagName: 'li',
            key: undefined,
            count: 1,
            props: {},
            children: ['Item #0']
          },
          {
            tagName: 'li',
            key: undefined,
            count: 1,
            props: {},
            children: ['Item #1']
          }
        ]
      }
    ]
  };

newChildren = diffs.children;
oldChildren數據以下:

oldChildren = [
    {
      tagName: 'h1',
      key: undefined
      count: 1
      props: {style: 'colod: red'},
      children: ['simple virtal dom']
    },
    {
      tagName: 'p',
      key: undefined
      count: 1
      props: {},
      children: ['the count is :1']
    },
    {
      tagName: 'ul',
      key: undefined
      count: 2
      props: {},
      children: [
        {
          tagName: 'li',
          key: undefined,
          count: 1,
          props: {},
          children: ['Item #0']
        }
      ]
    }
  ];

接着就是遍歷 oldChildren, 第一次遍歷時 leftNode 爲null,所以 currentNodeIndex = currentNodeIndex + 1 = 0 + 1 = 1; 不是第一次遍歷,那麼leftNode都爲上一次遍歷的子節點,所以不是第一次遍歷的話,那麼 currentNodeIndex = currentNodeIndex + leftNode.count + 1; 
而後遞歸調用 deepWalk(child, newChild, currentNodeIndex, patches); 方法,接着把child賦值給leftNode,leftNode = child;

因此一直遞歸遍歷,最終把不相同的節點 會存儲到 currentPatch 數組內。最後執行

if (currentPatch.length) {
   patches[index] = currentPatch;
}

把對應的currentPatch 存儲到 patches對象內中的對應項,最後就返回 patches對象。

4. 返回到index.js 代碼內,把兩顆不相同的樹節點的提取出來後,須要調用patch.js方法傳進;把不相同的節點應用到真正的DOM上.
不相同的節點 patches數據以下:

patches = {
    1: [{type: 2, props: {style: 'color: blue'}}],
    4: [{type: 3, content: 'the count is :2'}],
    5: [
          { 
              type: 1, 
              moves: [
                { index: 1, 
                   item: {
                      tagName: 'li',
                      props: {},
                      count: 1,
                      key: undefined,
                      children: ['Item #1']
                    }
                }
              ]
           }
        ]
    }

以下代碼調用:
patch(root, patches);
執行patch方法,代碼以下:

function patch(node, patches) {
  var walker = {index: 0};
  deepWalk(node, walker, patches);
}

deepWalk 代碼以下:

function deepWalk(node, walker, patches) {
   var currentPatches = patches[walker.index];
      // node.childNodes 返回指定元素的子元素集合,包括HTML節點,全部屬性,文本節點。
   var len = node.childNodes ? node.childNodes.length : 0;
   for (var i = 0; i < len; i++) {
      var child = node.childNodes[i];
      walker.index++;
      // 深度複製 遞歸遍歷
      deepWalk(child, walker, patches);
   }
   if (currentPatches) {
      applyPatches(node, currentPatches);
   }
}

1. 首次調用patch的方法,root就是container的節點,所以調用deepWalk方法,所以 var currentPatches = patches[0] = undefined,
var len = node.childNodes ? node.childNodes.length : 0; 所以 len = 3; 很明顯該子節點的長度爲3,由於子節點有 h1, p, 和ul元素;


2. 而後進行for循環,獲取該父節點的子節點,所以第一個子節點爲 h1 元素,walker.index++; 所以walker.index = 1; 再進行遞歸 deepWalk(child, walker, patches); 此時子節點爲h1, walker.index爲1, 所以獲取 currentPatches = patches[1]; 獲取值,再獲取 h1的子節點的長度,len = 1; 而後再for循環,獲取child爲文本節點,此時 walker.index++; 因此此時walker.index 爲2, 在調用deepwalk方法遞歸,所以再繼續獲取 currentPatches = patches[2]; 值爲undefined,再獲取len = 0; 由於文本節點麼有子節點,因此for循環跳出,因此判斷currentPatches是否有值,由於此時 currentPatches 爲undefined,因此遞歸結束,再返回到 h1元素上來,因此currentPatches = patches[1]; 因此有值,因此調用 applyPatches()方法來更新dom元素。


3. 繼續循環 i, 此時i = 1; 獲取子節點 child = p元素,walker.index++,此時walker.index = 3, 繼續調用 deepWalk方法,獲取 var currentPatches = patches[walker.index] = patches[3]的值,var len = 1; 由於p元素下有一個子節點(文本節點),再進for循環,此時 walker.index++; 所以walker.index = 4; child此時爲文本節點,在調用 deepwalk方法的時候,再獲取var currentPatches = patches[walker.index] = patches[4]; 再執行len 代碼的時候 len = 0;所以跳出for循環,判斷 currentPatches是否有值,有值的話,更新對應的DOM元素。

4. 繼續循環i = 2; 獲取子節點 child = ul元素,walker.index++; 此時walker.index = 5; 在調用deepWalk方法遞歸,所以再獲取 var currentPatches = patches[walker.index] = patches[5]; 而後len = 1, 由於ul元素下有一個li元素,在繼續for循環遍歷,獲取子節點li,此時walker.index++; walker.index = 6; 再遞歸調用deepwalk方法,再獲取var currentPatches = patches[walker.index] = patches[6]; len = 1; 由於li的元素下有一個文本節點,再進行for循環,此時child爲文本節點,walker.index++;此時walker.index = 7; 再執行 deepwalk方法,再獲取 var currentPatches = patches[walker.index] = patches[7]; 這時候 len = 0了,所以跳出for循環,判斷 當前的currentPatches是否有值,沒有,就跳出,而後再返回ul元素,獲取該本身li的時候,walker.index 等於5,所以var currentPatches = patches[walker.index] = patches[5]; 而後判斷 currentPatches是否有值,有值就進行更新DOM元素。

最後就是 applyPatches 方法更新dom元素了,以下代碼:

function applyPatches(node, currentPatches) {
  utils.each(currentPatches, function(currentPatch) {
    switch (currentPatch.type) {
      case REPLACE:
        var newNode = (typeof currentPatch.node === 'string') 
          ? document.createTextNode(currentPatch.node)
          : currentPatch.node.render();
        node.parentNode.replaceChild(newNode, node);
        break;
      case REORDER:
        reorderChildren(node, currentPatch.moves);
        break;
      case PROPS: 
        setProps(node, currentPatch.props);
        break;
      case TEXT:
        if(node.textContent) {
          node.textContent = currentPatch.content;
        } else {
          // ie bug
          node.nodeValue = currentPatch.content;
        }
        break;
      default:
        throw new Error('Unknow patch type' + currentPatch.type);
    }
  });
}

判斷類型,替換對應的屬性和節點。
最後就是對子節點進行排序的操做,代碼以下:

// 對子節點進行排序
function reorderChildren(node, moves) {
  var staticNodeList = utils.toArray(node.childNodes);
  var maps = {};
  utils.each(staticNodeList, function(node) {
    // 若是是元素節點
    if (node.nodeType === 1) {
      var key = node.getAttribute('key');
      if (key) {
        maps[key] = node;
      }
    }
  })
  utils.each(moves, function(move) {
    var index = move.index;
    if (move.type === 0) {
      // remove Item
      if (staticNodeList[index] === node.childNodes[index]) {
        node.removeChild(node.childNodes[index]);
      }
      staticNodeList.splice(index, 1);
    } else if(move.type === 1) {
      // insert item
      var insertNode = maps[move.item.key] 
        ? maps[move.item.key].cloneNode(true)
        : (typeof move.item === 'object') ? move.item.render() : document.createTextNode(move.item);
      staticNodeList.splice(index, 0, insertNode);
      node.insertBefore(insertNode, node.childNodes[index] || null);
    }
  });
}

遍歷moves,判斷moves.type 是等於0仍是等於1,等於0的話是刪除操做,等於1的話是新增操做。好比如今moves值變成以下:

moves = {
  index: 1,
  type: 1,
  item: {
    tagName: 'li',
    key: undefined,
    props: {},
    count: 1,
    children: ['#Item 1']
  }
};

node節點 就是 'ul'元素,var staticNodeList = utils.toArray(node.childNodes); 把ul的舊子節點li轉成Array形式,因爲沒有屬性key,因此直接跳到下面遍歷代碼來,遍歷moves,獲取某一項的索引index,判斷move.type 等於0 仍是等於1, 目前等於1,是新增一項,可是沒有key,所以調用move.item.render(); 渲染完後,對staticNodeList數組裏面的舊節點的li項從第二項開始插入節點li,而後執行node.insertBefore(insertNode, node.childNodes[index] || null); node就是ul父節點,insertNode節點插入到 node.childNodes[1]的前面。所以把在第二項的前面插入第一項。
查看github上源碼

相關文章
相關標籤/搜索