【虚拟DOM】浅析 虚拟DOM

简介: 【虚拟DOM】浅析 虚拟DOM

虚拟DOM作为目前流行的DOM操作思想,被广泛用在react中,这套设计的确在用户体验上带来了显著提升。下面我们来浅析一下这个东西,一步步看下去,希望你能有所收获。

图片.png

设计理念

尽管MVVM将页面逻辑实现的核心转移到数据层面的修改上,但是最终数据层反映到页面上View的层的渲染和改变仍是通过对应的指令进行DOM操作来完成的。而且,通常一次ViewModel的变化可能会触发液面上多个指令操作DOM的变化,从而造成页面结构层发生大量DOM操作或渲染。

例如:

图片.png

当你使用MVVM时,就会生成一个列表。

现在我们把 content 变成:

[{value:0},{value:1},{value:2},{value:3}],即增加一个。

如果在MVVM中一般会重新渲染整个列表,包括列表中无须改变的部分也会重新渲染一次,例如包含值1,2,3的三个列表。

但是,你肯定会想,其实只要直接改变DOM,在<ul>子元素前插入一个新的

<li>就OK了,是不是。

但是,通常MVVM是不会这么做的。

所以,当发生大量DOM操作时,会消耗更多性能。

那问题来了,该如何改进ViewModel呢?即,如何只增加一个<li>这个问题。

这里我们需要结合“新旧比较”的思想,将新的Model data和旧的Model data对比,然后记录ViewModel的改变方式和位置,就知道这次View层应该怎样更新,这样比直接重新渲染整个列表高效。

简单而言,ViewModel里的数据就是描述页面View内容的另一种数据结构,不过需要结合特定的MVVM描述语法编译生成完整的DOM结构。那么,结合后上面的代码就变成下面这样:

图片.png

此时我们需要增加一个新的<li>,按照上面的思路,就是先生成一个新的ulElement,然后与旧的ulElement进行结构上的对比,那么,其实就是在旧的ulElement对象的children属性额最前面增加一份内容,即:

{tagName:'li',
      children:[{
                 tagName:'span',
                 nodeValue:0
        },{
                 tagName:'span',
                 nodeText:'text-0'
 }]}

但是,你要注意它不是在旧的ulElement上增加,而是生成一个新的包含新增的ulElement

此时,你可以把这里的ulElement理解为VirtualDOM(虚拟DOM)。

虚拟DOM是什么?先来看一段定义:

VirtualDOM是一个能够直接描述一段HTML DOM结构的Javascript对象,浏览器可以根据其结构按照一定规则创建出确定唯一的HTML DOM结构。

下面我们具体讲解下虚拟DOM的核心实现思路。

图片.png

实现核心思路

从上一节中,我们稍微知道了什么是虚拟DOM,用一句话总结其操作的核心可分为三步:

  1. 创建Virtual DOM;
  2. 对比两个Virtual DOM 生成差异化VirtualDOM;
  3. 将差异化Virtual DOM 渲染到页面上;

下面我们从这三个步骤分别讲起。

创建虚拟 DOM

如何创建呢?你可能会想通过遍历HTML片段创建,但是这样创建有一个问题,因为遍历HTML就意味着你要使用到DOM的读取操作,那这样的话,不是就没有多大意义了。

一个更通用的方法是,自己实现HTML字符串文本的解析方式,根据标签之间的关系,读取生成Virtual DOM结构。例如:

图片.png

现在关键是createVDOM如何实现了。

我们再回头看看:

图片.png

根据上图,在createVDOM的实现上,需要逐个分析字符串htmlString_ul中的字符,根据词法分析内容,将标签名字存入tagName,属性存入attributes,子标签内容存入children

这样,你就将一段HTML字符串解析成一个Javascript对象了。

到现在,有没有更深刻地体会到什么不同之处了?

图片.png

我们看到,上面的方式是Javascript通过直接分析HTML字符串文本来生成VirtualDOM,而非对DOM API进行操作。

所以,小结一下,创建VirtualDOM的过程就是将一段DOM描述字符串解析成VirtualDOM对象的过程,这相当于实现了一个HTML文本解析器,但没有生成DOM对象树。故当交给浏览器解析的时候解析的不是HTML,而是Javascript对象。

但是。。还没完呢。生成了VirtualDOM后,还需要进行渲染,生成一个真实的DOM,毕竟前面的是虚拟的嘛。

来看一下如何渲染生成真实的DOM吧:

图片.png

接下来,进入第二步骤。

VirtualDOM新旧对比

图片.png

当发生改变时,通常会生成一个新的VirtualDOM结构来表示改变后的状态,然后进行“新旧”比较,找出差异性,得到一个差异树对象。

这里面有一个关键的地方,就是如何进行“新旧”比较。

这里用到的算法实际上是对多叉树结构的遍历算法。而该遍历算法又分为深度与广度遍历。这里我们主要以深度优先遍历算法来讲解“新旧”比较的过程。(广度优先类似)

图片.png

我们先看上图,对上图进行深度优先遍历的结果:abdecfg

接着改变一下,新增一块内容,如下图红色部分:

图片.png

此时,对该图进行深度优先遍历的结果:ahijbdecfg

我们把这两个图的深度优先遍历结果放在一起,如下:

图片.png

现在,我们是不是能容易地分析出需要再a和b节点之间插入hij节点。然后根据hij节点的关系,可以看出节点h是i与j节点的父节点,那么我们就知道了现在只需要插入完整的h节点。

上面有提到广度优先类似,稍微看一下吧:

广度优先的遍历结果分别如下:

图片.png

此时我们发现有2处是需要插入的,即需要进行2步操作,那么这就需要进一步判断来合并这2个操作。

小结一下,上面我们只是讲到了“新旧”对比,主要涉及如何发现其中改变的内容,实际上除了这些,你还需要记录发生差异化改变的类型和位置,例如是对哪一类元素进行增加\删除\替换操作等。

最后,第三步是渲染新生成的差异化虚拟DOM。

渲染新生成的差异化虚拟DOM

经过差异化比较后,你能获取到发生改变之后的“差异化VirtualDOM",”差异化类型“和”差异化位置“。

现在就很明朗了,剩下的操作就是将差异化内容经过DOM操作渲染到页面上即可完成。

总结一下,虚拟DOM最本质的区别是使用Javascript对象替代了DOM对象树,从而提升页面渲染性能。

图片.png

最后,祝大家五一节愉快地加班。。。

参考:w3c


END



相关文章
|
JavaScript 前端开发 编译器
说说你对虚拟 DOM 的理解?
说说你对虚拟 DOM 的理解?
111 0
|
10月前
|
JavaScript 前端开发 算法
React技术栈-虚拟DOM和DOM diff算法
这篇文章介绍了React技术栈中的虚拟DOM和DOM diff算法,并通过一个实际案例展示了如何使用React组件和状态管理来实现动态更新UI。
102 2
|
9月前
|
JavaScript 前端开发 算法
真实DOM和虚拟DOM有哪些区别?
本文介绍了真实DOM和虚拟DOM的概念、使用方式、优势、劣势、使用场景、影响因素、开发效率和性能对比。真实DOM是浏览器提供的原生接口,直接操作简单直观,但频繁操作会导致性能损耗。虚拟DOM是真实DOM的抽象表示,通过比较差异减少DOM操作,适用于大规模数据变更和复杂交互的页面。开发者应根据具体需求选择合适的DOM操作方式,以提高页面性能和开发效率。
405 1
真实DOM和虚拟DOM有哪些区别?
|
11月前
|
移动开发 JavaScript 前端开发
【Vue面试题二十二】、什么是虚拟DOM?如何实现一个虚拟DOM?说说你的思路
这篇文章深入探讨了虚拟DOM的概念、必要性以及在Vue中的实现方式,解释了虚拟DOM如何作为真实DOM的轻量级抽象,通过优化DOM操作提高性能,并实现跨平台渲染的能力。
【Vue面试题二十二】、什么是虚拟DOM?如何实现一个虚拟DOM?说说你的思路
|
11月前
|
JavaScript 算法 前端开发
"揭秘虚拟DOM的神奇魔法:从零开始打造高效渲染引擎,颠覆你的DOM操作认知!"
【8月更文挑战第20天】虚拟DOM是一种优化技术,通过在内存中构建DOM树的轻量级副本,减少浏览器重排和重绘,提升性能。本文简要介绍了DOM及其重要性,并深入解释了虚拟DOM的概念。虚拟DOM通过模拟真实DOM结构,在内存中进行数据更新,仅将变动部分同步到实际DOM。文中还提供了一个简易虚拟DOM的实现方案,包括虚拟节点创建、渲染函数及一个基本的diff算法,用于比对新旧虚拟DOM并高效更新实际DOM。通过示例展示了如何构建和渲染一个简单的虚拟DOM。这有助于理解虚拟DOM的基本工作原理和技术细节。
113 4
|
11月前
|
JavaScript 前端开发 API
区分 DOM 与虚拟 DOM
【8月更文挑战第24天】
264 0
|
JavaScript 前端开发 算法
虚拟DOM是React的关键技术,它是个轻量的JS对象树,模拟实际DOM结构。
【6月更文挑战第27天】虚拟DOM是React的关键技术,它是个轻量的JS对象树,模拟实际DOM结构。当状态改变,React不直接修改DOM,而是先构建新的虚拟DOM树。通过 diff 算法比较新旧树,找到最小变更,仅更新必要部分,提高性能,避免频繁DOM操作。虚拟DOM还支持跨平台应用,如React Native。它优化了更新流程,简化开发,并提升了用户体验。
110 1
|
JavaScript 算法 前端开发
虚拟Dom
虚拟Dom
|
JavaScript 前端开发 算法
深入理解虚拟DOM:原理、优势与实践
深入理解虚拟DOM:原理、优势与实践
1114 0
|
JavaScript 算法 前端开发
Vue的虚拟DOM:Vue虚拟DOM的工作原理
【4月更文挑战第24天】Vue的虚拟DOM提升渲染性能,通过创建JavaScript对象树(虚拟DOM树)来跟踪DOM变化。当状态改变,Vue用新的虚拟DOM树与旧树对比(diff算法),找到最小DOM操作集合来更新真实DOM。优化策略包括减少状态变化、使用key属性和简化组件结构。理解虚拟DOM工作原理有助于Vue的性能优化。

热门文章

最新文章