QCon北京「鸿蒙专场」火热来袭!即刻报名,与创新同行~ 了解详情
写点什么

你不知道的 virtual DOM(二):Virtual Dom 的更新

  • 2020-03-08
  • 本文字数:3830 字

    阅读完需:约 13 分钟

你不知道的virtual DOM(二):Virtual Dom的更新

一、前言

目前最流行的两大前端框架,React 和 Vue,都不约而同的借助 Virtual DOM 技术提高页面的渲染效率。那么,什么是 Virtual DOM ?它是通过什么方式去提升页面渲染效率的呢?本系列文章会详细讲解 Virtual DOM 的创建过程,并实现一个简单的 Diff 算法来更新页面。本文的内容脱离于任何的前端框架,只讲最纯粹的 Virtual DOM 。敲单词太累了,下文 Virtual DOM 一律用 VD 表示。


这是 VD 系列文章的第二篇,本文将会实现一个简单的 VD Diff 算法,计算出差异并反映到真实的 DOM 上去。

二、思路

使用 VD 的框架,一般的设计思路都是页面等于页面状态的映射,即 UI=render(state)。当需要更新页面的时候,无需关心 DOM 具体的变换方式,只需要改变 state即可,剩下的事情( render)将由框架代劳。我们考虑最简单的情况,当 state 发生变化时,我们重新生成整个 VD ,触发比较的操作。上述过程分为以下四步:


  • state 变化,生成新的 VD

  • 比较 VD 与之前 VD 的异同

  • 生成差异对象( patch

  • 遍历差异对象并更新 DOM 差异对象的数据结构是下面这个样子,与每一个 VDOM 元素一一对应:


{  type,  vdom,  props: [{        type,        key,        value       }]  children}
复制代码


最外层的 type 对应的是 DOM 元素的变化类型,有 4 种:新建、删除、替换和更新。props 变化的 type 只有 2 种:更新和删除。枚举值如下:


const nodePatchTypes = {  CREATE: 'create node',  REMOVE: 'remove node',  REPLACE: 'replace node',  UPDATE: 'update node'}
const propPatchTypes = { REMOVE: 'remove prop', UPDATE: 'update prop'}
复制代码

三、代码实现

我们做一个定时器,500 毫秒运行一次,每次对 state 加 1。页面的 li元素的数量随着 state 而变。


let state = { num: 5 };let timer;let preVDom;
function render(element) { // 初始化的 VD const vdom = view(); preVDom = vdom;
const dom = createElement(vdom); element.appendChild(dom);

timer = setInterval(() => { state.num += 1; tick(element); }, 500);}
function tick(element) { if (state.num > 20) { clearTimeout(timer); return; }
const newVDom = view();}
function view() { return ( <div> Hello World <ul> { // 生成元素为0到n-1的数组 [...Array(state.num).keys()] .map( i => ( <li id={i} class={`li-${i}`}> 第{i * state.num} </li> )) } </ul> </div> );}
复制代码


接下来,通过对比 2 个 VD,生成差异对象。


function tick(element) {  if (state.num > 20) {    clearTimeout(timer);    return;  }
const newVDom = view();
// 生成差异对象 const patchObj = diff(preVDom, newVDom);}
function diff(oldVDom, newVDom) { // 新建 node if (oldVDom == undefined) { return { type: nodePatchTypes.CREATE, vdom: newVDom } }
// 删除 node if (newVDom == undefined) { return { type: nodePatchTypes.REMOVE } }
// 替换 node if ( typeof oldVDom !== typeof newVDom || ((typeof oldVDom === 'string' || typeof oldVDom === 'number') && oldVDom !== newVDom) || oldVDom.tag !== newVDom.tag ) { return { type: nodePatchTypes.REPLACE, vdom: newVDom } }
// 更新 node if (oldVDom.tag) { // 比较 props 的变化 const propsDiff = diffProps(oldVDom, newVDom);
// 比较 children 的变化 const childrenDiff = diffChildren(oldVDom, newVDom);
// 如果 props 或者 children 有变化,才需要更新 if (propsDiff.length > 0 || childrenDiff.some( patchObj => (patchObj !== undefined) )) { return { type: nodePatchTypes.UPDATE, props: propsDiff, children: childrenDiff } }
}}
// 比较 props 的变化function diffProps(oldVDom, newVDom) { const patches = [];
const allProps = {...oldVDom.props, ...newVDom.props};
// 获取新旧所有属性名后,再逐一判断新旧属性值 Object.keys(allProps).forEach((key) => { const oldValue = oldVDom.props[key]; const newValue = newVDom.props[key];
// 删除属性 if (newValue == undefined) { patches.push({ type: propPatchTypes.REMOVE, key }); } // 更新属性 else if (oldValue == undefined || oldValue !== newValue) { patches.push({ type: propPatchTypes.UPDATE, key, value: newValue }); } } )
return patches;}
// 比较 children 的变化function diffChildren(oldVDom, newVDom) { const patches = [];
// 获取子元素最大长度 const childLength = Math.max(oldVDom.children.length, newVDom.children.length);
// 遍历并diff子元素 for (let i = 0; i < childLength; i++) { patches.push(diff(oldVDom.children[i], newVDom.children[i])); }
return patches;}
复制代码


计算得出的差异对象是这个样子的:


{  type: "update node",  props: [],  children: [    null,     {      type: "update node",      props: [],      children: [        null,         {          type: "update node",          props: [],          children: [            null,             {              type: "replace node",              vdom: 6            }          ]        }      ]    },    {      type: "create node",      vdom: {        tag: "li",        props: {          id: 5,          class: "li-5"        },        children: ["第", 30]      }    }  ]}
复制代码


下一步就是遍历差异对象并更新 DOM 了:


function tick(element) {  if (state.num > 20) {    clearTimeout(timer);    return;  }
const newVDom = view();
// 生成差异对象 const patchObj = diff(preVDom, newVDom);
preVDom = newVDom;
// 给 DOM 打个补丁 patch(element, patchObj);}
// 给 DOM 打个补丁function patch(parent, patchObj, index=0) { if (!patchObj) { return; }
// 新建元素 if (patchObj.type === nodePatchTypes.CREATE) { return parent.appendChild(createElement(patchObj.vdom)); }
const element = parent.childNodes[index];
// 删除元素 if (patchObj.type === nodePatchTypes.REMOVE) { return parent.removeChild(element); }
// 替换元素 if (patchObj.type === nodePatchTypes.REPLACE) { return parent.replaceChild(createElement(patchObj.vdom), element); }
// 更新元素 if (patchObj.type === nodePatchTypes.UPDATE) { const {props, children} = patchObj;
// 更新属性 patchProps(element, props);
// 更新子元素 children.forEach( (patchObj, i) => { // 更新子元素时,需要将子元素的序号传入 patch(element, patchObj, i) }); }}
// 更新属性function patchProps(element, props) { if (!props) { return; }
props.forEach( patchObj => { // 删除属性 if (patchObj.type === propPatchTypes.REMOVE) { element.removeAttribute(patchObj.key); } // 更新或新建属性 else if (patchObj.type === propPatchTypes.UPDATE) { element.setAttribute(patchObj.key, patchObj.value); } })}
复制代码


到此为止,整个更新的流程就执行完了。可以看到页面跟我们预期的一样,每 500 毫秒刷新一次,构造渲染树和绘制页面花的时间也非常少。



作为对比,如果我们在生成新的 VD 后,不经过比较,而是直接重新渲染整个 DOM 的时候,会怎样呢?我们修改一下代码:


function tick(element) {  if (state.num > 20) {    clearTimeout(timer);    return;  }
const newVDom = view(); newDom = createElement(newVDom);
element.replaceChild(newDom, dom);
dom = newDom;
/* // 生成差异对象 const patchObj = diff(preVDom, newVDom);
preVDom = newVDom;
// 给 DOM 打个补丁 patch(element, patchObj); */}
复制代码


效果如下:



可以看到,构造渲染树( Rendering)和绘制页面( Painting)的时间要多一些。但另一方面花在 JS 计算( Scripting)的时间要少一些,因为不需要比较节点的变化。如果算总时间的话,重新渲染整个 DOM 花费的时间反而更少,这是为什么呢?


其实原因很简单,因为我们的 DOM 树太简单了!节点很少,使用到的 css 也很少,所以构造渲染树和绘制页面就花不了多少时间。VD 真正的效果还是要在真实的项目中才体现得出来。

四、总结

本文详细介绍如何实现一个简单的 VD Diff 算法,再根据计算出的差异去更新真实的 DOM 。然后对性能做了一个简单的分析,得出使用 VD 在减少渲染时间的同时增加了 JS 计算时间的结论。基于当前这个版本的代码还能做怎样的优化呢,请期待下一篇的内容:《你不知道的 Virtual DOM(三):Virtual DOM 更新优化》


P.S: 想看完整代码见这里,如果有必要建一个仓库的话请留言给我:代码(https://gist.github.com/dickenslian/a0a8d41a88d566d86271de16cd7738f0


2020-03-08 19:241571

评论

发布
暂无评论
发现更多内容

超越传统:人工智能赋能的自动化测试新前景

测吧(北京)科技有限公司

测试

Web3.0区块链技术全流程方案:DApp项目开发、推广以及运营

区块链软件开发推广运营

dapp开发 区块链开发 链游开发 NFT开发 公链开发

亚马逊云科技为派拓网络打造数字安全平台产品组合

财见

前端:Vue2.0和Vue3.0的一些入门对比

秃头小帅oi

前端 低代码 js Vue 3 vue2

蒋安祥:巴拿马奖项评选指南与规程赢得企业界广泛认可

极客天地

SD-WAN网络搭建技术:企业降本增效的首选

Ogcloud

SD-WAN 企业网络 SD-WAN组网 SD-WAN服务商 SDWAN

云桌面系统厂家-青椒云

青椒云云电脑

云桌面 云桌面厂家 云桌面方案 云桌面系统

商品评价聚合:利用API从多个来源获取数据的详细指南

Noah

蜗牛游戏宣布利用AI技术提升其开发流程

财见

作为程序员,沟通能力是否重要?

小齐写代码

EMQX Enterprise 5.5 发布:新增 Elasticsearch 数据集成

EMQ映云科技

mqtt mqtt broker

教你如何用Keepalived和HAproxy配置高可用 Kubernetes 集群

华为云开发者联盟

开发 华为云 华为云开发者联盟

舞台LED显示屏与传统LED显示屏的区别

Dylan

LED显示屏 全彩LED显示屏 led显示屏厂家 户内led显示屏 舞台表演

云桌面跟远程桌面有什么区别?

青椒云云电脑

云桌面 云桌面方案 云桌面系统

PTS 3.0:开启智能化的压测瓶颈分析

阿里巴巴云原生

阿里云 云原生 压测

火山引擎弹性容器实例:从节点中心转型 Serverless 化架构的利器

极客天地

陆海×微帧,在海洋卫星传输环境下的极限视频压缩

微帧Visionular

视频编码 视频压缩

SD-WAN对金融行业的重要性

Ogcloud

SD-WAN 企业网络 SD-WAN组网 SD-WAN服务商 SDWAN

Supermicro 通过新基础设施解决方案,加速 5G 和电信云工作负载性能

财见

追踪Jira中项目成本与工时,更符合国人使用习惯——TimeWise工时管理

龙智—DevSecOps解决方案

DevOps DevSecOps Atlassian

破防了,谁懂啊家人们:记一次mysql问题排查

阿里技术

MySQL 索引 问题排查 表结构

聊聊Java 类属性与类方法的应用

伤感汤姆布利柏

Java 前端

开班在即 | 测试开发名企定向培养训练营,手把手带你提升核心竞争力!

测吧(北京)科技有限公司

测试

KaiwuDB 拿下 “物联之星” 双项殊荣

KaiwuDB

数据库 物联网

Udemy 上最受欢迎的免费编程课程

秃头小帅oi

php 学习 React 课程 java

SNZ资本的首席信息官Gavin确认出席Hack .Summit() 2024香港开发者大会!

TechubNews

云桌面哪家好用?

青椒云云电脑

云桌面 云桌面厂家 云桌面解决方案

聚道云助力时尚巨头打通数据孤岛,实现全渠道管理升级!

聚道云软件连接器

案例分享

你不知道的virtual DOM(二):Virtual Dom的更新_文化 & 方法_大白_InfoQ精选文章