2021-03-11 dom-diff - 03

patch.js

// patch.js 几种对比方案
/**
 * 比较两个虚拟节点的key和type是否相同
 * @param {Vnode} oldVnode 旧节点 
 * @param {Vnode} newVnode 新节点
 * @returns {boolean} 返回两个节点的key和type是否都是一致
 */
function isSameVnode(oldVnode, newVnode) {
  return oldVnode.key === newVnode.key && oldVnode.type === newVnode.type;
}

// 需要一个方法, 做成一个映射表: { a: 0, b: 1, c: 2, d: 3 }
/**
 * 将节点的key和索引做成一个映射表
 * @param {Array<vnode>} oldChildren 节点数组
 * @returns {Map} 返回节点键值和索引的映射map
 */
function createMapByKeyToIndex(oldChildren) {
  const map = {};
  for (let i = 0; i < oldChildren.length; i += 1) {
    const current = oldChildren[i];
    if (current.key) {
      map[current.key] = i;
    }
  }
  return map;
}

/**
 * 对比新旧子节点, 并在parent真实节点中进行替换
 * @param {HTMLElement} parent 父真实节点
 * @param {Array<vnode>} oldChildren 旧的children虚拟节点
 * @param {Array<vnode>} newChildren 新的children虚拟节点
 */
function updateChildren(parent, oldChildren, newChildren)  {
  // 最复杂的就是列表
  /**
    h('div', {}, 
      h('li', { style: { color: "red" }, key: 'A' }, 'A'),
      h('li', { style: { color: "yellow" }, key: 'B' }, 'B'),
      h('li', { style: { color: "blue" }, key: 'C' }, 'C'),
      h('li', { style: { color: "green" }, key: 'D' }, 'D')
    )
    => 更新后
    h('div', {}, 
      h('li', { style: { color: "yellow" }, key: 'A', id: 'a1' }, 'A1'),
      h('li', { style: { color: "blue" }, key: 'B' }, 'B1'),
      h('li', { style: { color: "green" }, key: 'C' }, 'C1'),
      h('li', { style: { color: "red" }, key: 'D' }, 'D1'),
      h('li', { style: { color: "orange" }, key: 'E' }, 'E1')
    )
  
  **/

  // 对常见的dom 操作做优化
  // 1. 前后, 追加 push, unshift
  // 2. 正序倒序
  // 3. 表示头部有新节点, 需要追加在头部
  // 4. 暴力对比, 节点复用

  // 旧节点 头部指针
  let oldStartIndex = 0;
  // 旧节点 头部节点
  let oldStartVnode = oldChildren[oldStartIndex];

  // 旧节点 尾部指针
  let oldEndIndex = oldChildren.length - 1;
  // 旧节点 尾部节点
  let oldEndVnode = oldChildren[oldEndIndex];

  // 旧节点 头部指针
  let newStartIndex = 0;
  // 旧节点 头部节点
  let newStartVnode = newChildren[newStartIndex];

  // 旧节点 尾部指针
  let newEndIndex = newChildren.length - 1;
  // 旧节点 尾部节点
  let newEndVnode = newChildren[newEndIndex];

  // 旧节点的键值和索引映射对象
  const oldChildrenKeyIndexMap = createMapByKeyToIndex(oldChildren);

  // 1. 判断 oldChildren和newChildren 循环的时候, 谁先结束就停止循环
  while (oldStartIndex <= oldEndIndex && newStartIndex <= newEndIndex) {
    // 表示节点被移除之后, 如果为undefined, 直接跳过该节点
    if (!oldStartVnode) {
      oldStartVnode = oldChildren[++newStartIndex];
    } else if (!oldEndVnode) {
      oldEndVnode = oldChildren[--oldEndIndex];
    }

    // 判断是否是同一个节点, 从头部开始比较
    else if (isSameVnode(oldStartVnode, newStartVnode)) {
      patch(oldStartVnode, newStartVnode); // 主要是做属性更新的操作和子文本节点
      oldStartIndex += 1;
      newStartIndex += 1;
      oldStartVnode = oldChildren[oldStartIndex];
      newStartVnode = newChildren[newStartIndex];
    } else if (isSameVnode(oldEndVnode, newEndVnode)) {
      // 如果头部不相同, 从尾部开始比较
      // A B C D
      // D C B A
      patch(oldEndVnode, newEndVnode);
      oldEndIndex -= 1;
      newEndIndex -= 1;
      oldEndVnode = oldChildren[oldEndIndex];
      newEndVnode = newChildren[newEndIndex];
    } else if (isSameVnode(oldStartVnode, newEndVnode)) {
      // 如果老的头部和新的尾部是相同的节点
      // A B C D
      // B C D A
      patch(oldStartVnode, newEndVnode);
      parent.insertBefore(
        oldStartVnode.domElement,
        oldEndVnode.domElement.nextSiblings
      );
      oldStartIndex += 1;
      oldStartVnode = oldChildren[oldStartIndex];
      newEndIndex -= 1;
      newEndVnode = newChildren[newEndIndex];
    } else if (isSameVnode(oldEndVnode, newStartVnode)) {
      // 旧的尾部节点和新的头部节点相等
      // A B C D
      // D A B C
      patch(oldEndVnode, newStartVnode);
      parent.insertBefore(oldEndVnode.domElement, oldStartVnode.domElement); // 将老节点插入到了最前面
      oldEndIndex -= 1;
      oldEndVnode = oldChildren[oldEndIndex];
      newStartIndex += 1;
      newStartVnode = newChildren[newStartIndex];
    } else {
      // 暴力对比 (尽量可能的节点复用)
      // A B C D
      // G C A E F

      // 需要先拿到新的节点, 去老的节点中查找, 看是否存在, 如果存在就复用, 如果不存在就创建插入复用即可
      const index = oldChildrenKeyIndexMap[newStartVnode.key];
      if (index == null) {
        // 在新的对列中, 将当前旧节点插入到指定位置  (比如上面的G节点, 在旧节点数组中不存在, 那么就需要再旧节点数组中插入)
        parent.insertBefore(
          createDomElementFromVnode(newStartVnode),
          oldStartVnode.domElement
        );
      } else {
        // 表示节点在oldChildren存在, 此时需要复用 (比如上面的C节点, 需要将C节点移动到A节点的前面)
        const toMoveNode = oldChildren[index];
        // 需要移动的节点和新数组的开始节点比较
        patch(toMoveNode, newStartVnode);
        parent.insertBefore(toMoveNode.domElement, oldStartVnode.domElement);
        // 移动完成之后, 需要清除该节点
        oldChildren[index] = undefined;
      }
      newStartIndex += 1;
      newStartVnode = newChildren[newStartIndex];
    } // end 暴力对比
  }

  // 2. 表示有新的节点, 需要追加在尾部
  if (newStartIndex <= newEndIndex) {
    for (let i = newStartIndex; i <= newEndIndex; i += 1) {
      // const appendElement = createDomElementFromVnode(newChildren[i]);
      // parent.appendChild(appendElement);

      // 头部插入时: newEndIndex表示当前新节点中 [A,B,C,D] => [E,F,A,B,C,D]
      //      A,B,C,D
      //    F,A,B,C,D
      // newEndIndex代表的是 新节点中A的索引为1, newStartIndex此时仍然为0
      const beforeVnode = newChildren[newEndIndex + 1];
      const beforeElement = null;
      if (!beforeVnode) {
        // 表示是尾部追加
      } else {
        // 表示是头部追加
        beforeElement = beforeVnode.domElement;
      }

      // 主要考虑的是 头部比较和尾部比较的添加问题
      parent.insertBefore(
        createDomElementFromVnode(newChildren[i]),
        beforeElement
      );
    }
  }

  // 表示需要清除孩子节点
  // A B C D
  // G C A E F
  // G C A E F B D 需要删除尾部的BD节点
  if (oldStartIndex <= oldEndIndex) {
    for (let i = oldStartIndex; i <= oldEndIndex; i += 1) {
      if (oldChildren[i]) {
        parent.removeChild(oldChildren[i].domElement);
      }
    }
  }

}

测试

// 1. 先实现虚拟dom, 主要就是一个对象, 来表述dom节点, jsx
// createElement, h

import { h, patch, render } from './vdom/index';

/*
const vnode = h(
  "div",
  {
    key: "keyCode",
    id: "container",
  },
  h("span", { style: { color: "red" } }, "hello"),
  "dom diff"
);
*/
const list1 = h('ul',
  {}, 
  h('li', { style: { color: "red" }, key: 'A' }, 'A'),
  h('li', { style: { color: "yellow" }, key: 'B' }, 'B'),
  h('li', { style: { color: "blue" }, key: 'C' }, 'C'),
  h('li', { style: { color: "green" }, key: 'D' }, 'D')
);
const list2 = h(
  "ul",
  {},
  // 头部添加数据的时候, 从尾部开始比较
  h("li", { style: { color: "orange" }, key: "Z", id: "z1" }, "Z1"),
  h("li", { style: { color: "yellow" }, key: "A", id: "a1" }, "A1"),
  h("li", { style: { color: "blue" }, key: "B" }, "B1"),
  h("li", { style: { color: "green" }, key: "C" }, "C1"),
  h("li", { style: { color: "red" }, key: "D" }, "D1")
  // 尾部添加数据的时候, 从头部开始比较
  // h("li", { style: { color: "orange" }, key: "E" }, "E1")
);

// 暴力diff
// A B C D
// G C A E F
const list3 = h(
  "ul",
  {},
  h("li", { style: { color: "yellow" }, key: "A", id: "a1" }, "A"),
  h("li", { style: { color: "blue" }, key: "B" }, "B"),
  h("li", { style: { color: "green" }, key: "C" }, "C1"),
  h("li", { style: { color: "red" }, key: "D" }, "D1")
);
const list4 = h(
  "ul",
  {},
  h("li", { style: { color: "blue" }, key: "G" }, "G"),
  h("li", { style: { color: "green" }, key: "C" }, "C1"),
  h("li", { style: { color: "yellow" }, key: "A", id: "a1" }, "A1"),
  h("li", { style: { color: "red" }, key: "E" }, "E"),
  h("li", { style: { color: "red" }, key: "F" }, "F")
);

// list3 -> list4 => 结果 G C A1 E F


// 渲染 render: 将虚拟节点转换成为真实的dom节点, 最后插入到app元素中
// render(vnode, document.getElementById('app'))
render(list3, document.getElementById("app"));
// render(list1, document.getElementById("app"));

// 来一个新的节点 替换老的节点
// const newVnode = h('h2', {}, 'hello world');

setTimeout(() => {
  // 比对新老节点
  // patch(list1, list2);
  patch(list3, list4);
}, 2000);

// $mount

/**
 * <div id="container"><span style="color:red">hello</span>dom diff</div>
    h(
      'div',
      {
        key: 'keyCode',
        id: 'container'
      },
      h(
        'span',
        { style: { color: 'red' } },
        'hello'
      ),
      'dom diff'
    )
    
    生成的虚拟节点对象
    {
      type: 'div',
      props: { id: 'container' },
      children: [
        {
          type: 'span',
          props: {
            style: {
              color: red
            }
          },
          children: [],
          text: 'hello'
        },
        {
          type: '',
          props: {},
          children: [],
          text: 'dom diff'
        }
      ]
    }
 */

/**
用索引作为key的问题
A[checked=true] B C D E  => 首部添加F节点
F的key和type与A相同, 会直接复用A节点
F[checked=true] A B C D E
 */

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 162,825评论 4 377
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 68,887评论 2 308
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 112,425评论 0 255
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,801评论 0 224
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 53,252评论 3 299
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 41,089评论 1 226
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 32,216评论 2 322
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 31,005评论 0 215
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,747评论 1 250
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,883评论 2 255
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,354评论 1 265
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,694评论 3 265
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,406评论 3 246
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,222评论 0 9
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,996评论 0 201
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 36,242评论 2 287
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 36,017评论 2 281

推荐阅读更多精彩内容

  • patch的实现 调用
    FConfidence阅读 207评论 0 0
  • 2021.3.10 周四 小雨 体验课已经接近尾声,坚持约课。最后几天的话术发出后,回复的人反而多了起来...
    2f945987fa32阅读 90评论 0 0
  • 京❤️达总店:谢波 2021年3月1日 落地真经严格就是爱,放纵既是害 目标确认 目标: 今日体验: 关注库房油品...
    谢波1阅读 152评论 0 0
  • 我今天也没有洗脚。现在就是焦虑。特别焦虑。不知道要怎么办。也不困,睡不着。晚上吃饭的时候又被问省妇幼的事了。大家还...
    风知我意阅读 135评论 0 0
  • 推荐指数: 6.0 书籍主旨关键词:特权、焦点、注意力、语言联想、情景联想 观点: 1.统计学现在叫数据分析,社会...
    Jenaral阅读 5,662评论 0 5