秒懂Vue.jsDiff算法与虚拟DOM

一:虚拟DOM

虚拟DOM模仿了实际的 DOM 节点,并包含有关节点的信息,例如标签名、属性、子节点等。

简单举个例子:

js 复制代码
<ul>
  <li>1</li>
  <li>2</li>
  <li>3</li>
</ul>

Obj = {  // 虚拟dom结构, 根据虚拟dom生成真实dom
  tag: 'ul',
  children: [
    {tag: 'li', children: [{vnode: {text: '1'}}]},
    {tag: 'li', children: [{vnode: {text: '2'}}]},
  ]
}

在运行时,会把代码编译成一个对象,也就是虚拟 DOM,然后生成真实的 DOM,但在实际中,Vue 使用了一种更加复杂的对象结构来存储虚拟节点(VNode),并且包含了更多的元信息,比如节点的类型、属性、事件监听器等等。

二:diff算法

DOM结构发生改变时,会生成一个新的虚拟DOM,此时就需要一个算法来比较新旧DOM的不同来更快速的渲染,也就是diff算法。

比较过程中,会根据虚拟DOM树的结构,一层一层比较,首先比较的是两个虚拟 DOM的根节点是否相同,如果根节点相同才会继续检查子节点,看是否有可复用的DOM,否则不会再继续比较子节点。

那么会有一个疑问了,如果两个虚拟DOM只有根节点不同,子节点完全一样呢?确实,这样的两个新旧虚拟DOM树可复用高,但是如果 diff算法改成了不依据根节点是否相同来确定是否要检查子节点,那么会对逐一对比子节点找可复用部分,加大了引擎压力,有点捡了芝麻丢了西瓜那味了

2.1.判断两个跟节点是不是相同

  1. 判断两个节点上的key是否相同
  2. 判断两个节点的标签名是否相同
  3. 标签上的属性

2.2.diff比较过程

  1. 旧首节点,新首节点,旧尾结点,新尾结点
  2. 不是相同的节点,直接销毁 oldVnode,启用 newVnode
  3. newVnode是不是文字节点 (直接让浏览器修改文字)
  4. 有新的children 没有旧children (记录新增)
  5. 没有新的children 有旧children (记录删除)
  6. 有新的children 有旧childrendiff 核心工作 两两节点比较)

三:diff中的key

3.1.key为什么不能是index?

diff 中判断两个节点是否相同首先判断的就是两个节点的key是否相同, 如果用index作为keyindex不会随着元素位置的变更而移动,从而导致相同可复用的节点被认为不相同,降低了DOM的复用性,浪费性能

3.2.可不可以用随机数做key?

当然不可以,和index的原因是一样的,但比index效率更低下。用随机数的话新旧树的每一个DOM结构都是不同的key,用index,还可能有些DOM结构在新树中index没变,然后得以复用。

四:虚拟DOM的好处

想必友友们在其他文章中已经看过无数遍这个问题的答案了-->因为虚拟DOM性能好。

其实不然,虚拟DOM实际就是把HTML读成JS对象,再生成HTML结构,用虚拟DOM性能并不会有所提高,会开销很多v8的性能。一个新的前端框架svelte就是主打的无虚拟DOM,此外连尤雨溪自己都说在VUE4会把虚拟DOM砍掉。

优点:

  1. 跨平台:支持构建可以在多个平台上运行的应用程序,虚拟 DOM 允许开发者使用一套统一的 API(JavaScript)去描述用户界面,而不需要关心底层的具体实现细节,这在一定程度上促进了跨平台开发。
  2. 分摊了浏览器渲染线程的压力,减少了回流重绘 :当数据发生变化时,diff 算法能够计算出最小化的 DOM 更新操作,这样只需要对实际发生变化的部分进行更新。避免了不必要的 DOM 操作,从而减少了浏览器的回流和重绘,提高页面的响应速度和流畅度。
相关推荐
妙码生花7 分钟前
从 PHP 到 AI + Golang,程序员自救转型手记(十六):目录结构更新、完善 token 系统(AI 表示 token 入库无需加密?)
前端·后端·ai编程
程序me12 分钟前
Prompt、Context、Harness、Loop 之后是什么? AI工程下一个半年的关键词
前端·后端·ai编程
飞天狗36 分钟前
线上Bug一直复现不了?我用Sentry把错误追踪效率提升了10倍
前端
Slice_cy38 分钟前
对前端工程化的理解
前端
Slice_cy39 分钟前
状态机设计理念与实现
前端
星栈40 分钟前
LiveView 的生命周期:mount、handle_event 和 Socket 到底怎么运转
前端·前端框架·elixir
yingyima44 分钟前
JWT Token 解析与安全实践速查:5 问 5 答直击要害
前端
kyriewen2 小时前
我用 Codex 重写了同事维护三年的代码,他没说谢谢——而是找了领导
前端·javascript·ai编程
OpenTiny社区2 小时前
从零开发 AI 聊天页要两周?试试这款 Vue3 垂直对话组件库 TinyRobot,直接开箱即用
前端·vue.js·github
铁皮饭盒2 小时前
S3已成为文件存储标准,阿里/腾讯/华为云都支持,Bun率先原生支持
前端·javascript·后端