当前位置: 首页 > news >正文

Vue Diff算法原理深度解析:如何高效更新虚拟DOM?

在这里插入图片描述

文章目录

    • 1. 为什么需要Diff算法?
    • 2. Diff算法核心原则
    • 3. 核心流程图解
    • 4. 核心代码实现(简化版)
    • 5. Key的重要性示例
    • 6. 算法优化策略
    • 7. 时间复杂度优化
    • 8. 与其他框架的对比
    • 9. 总结

1. 为什么需要Diff算法?

在Vue的响应式系统中,当数据变化时,组件需要重新渲染。直接操作真实DOM非常消耗性能,因此Vue使用虚拟DOM(Virtual DOM)作为中间层。Diff算法的核心作用就是通过对比新旧虚拟DOM树,找出最小变更并批量更新真实DOM。

2. Diff算法核心原则

  • 同层比较:只比较同一层次的节点,不跨层级
  • 双端对比:同时从新旧子节点的首尾向中间扫描
  • 就地复用:通过key标识尽可能复用相同节点

3. 核心流程图解

相同
不同
相同
不同
相同
不同
相同
不同
开始对比
旧开始 vs 新开始?
更新节点并右移指针
旧结束 vs 新结束?
更新节点并左移指针
旧开始 vs 新结束?
移动节点到末尾
旧结束 vs 新开始?
移动节点到开头
查找Key索引
存在相同Key?
复用节点并移动
创建新节点

4. 核心代码实现(简化版)

function updateChildren(parentElm, oldCh, newCh) {let oldStartIdx = 0let oldEndIdx = oldCh.length - 1let newStartIdx = 0let newEndIdx = newCh.length - 1while (oldStartIdx <= oldEndIdx && newStartIdx <= newEndIdx) {// 四种对比情况if (sameVnode(oldCh[oldStartIdx], newCh[newStartIdx])) {patchVnode(oldCh[oldStartIdx], newCh[newStartIdx])oldStartIdx++newStartIdx++} else if (sameVnode(oldCh[oldEndIdx], newCh[newEndIdx])) {patchVnode(oldCh[oldEndIdx], newCh[newEndIdx])oldEndIdx--newEndIdx--}else if (sameVnode(oldCh[oldStartIdx], newCh[newEndIdx])) {// 移动节点到旧结束节点之后parentElm.insertBefore(oldCh[oldStartIdx].elm, oldCh[oldEndIdx].elm.nextSibling)patchVnode(oldCh[oldStartIdx], newCh[newEndIdx])oldStartIdx++newEndIdx--}else if (sameVnode(oldCh[oldEndIdx], newCh[newStartIdx])) {// 移动节点到旧开始节点之前parentElm.insertBefore(oldCh[oldEndIdx].elm, oldCh[oldStartIdx].elm)patchVnode(oldCh[oldEndIdx], newCh[newStartIdx])oldEndIdx--newStartIdx++}else {// Key查找逻辑const keyMap = createKeyMap(newCh, newStartIdx, newEndIdx)const idxInOld = findIdxInOld(oldCh, newStartVnode, keyMap)if (idxInOld) {// 移动已有节点parentElm.insertBefore(oldCh[idxInOld].elm, oldStartVnode.elm)patchVnode(oldCh[idxInOld], newCh[newStartIdx])oldCh[idxInOld] = undefined} else {// 创建新节点parentElm.insertBefore(createElm(newCh[newStartIdx]), oldStartVnode.elm)}newStartIdx++}}// 处理剩余节点if (oldStartIdx > oldEndIdx) {addNewNodes(parentElm, newCh, newStartIdx, newEndIdx)} else {removeOldNodes(parentElm, oldCh, oldStartIdx, oldEndIdx)}
}

5. Key的重要性示例

<!-- 没有key的情况 -->
<ul><li v-for="item in list">{{ item }}</li>
</ul><!-- 有key的情况 -->
<ul><li v-for="item in list" :key="item.id">{{ item.text }}</li>
</ul>

无Key时的Diff行为

  • 默认使用"就地复用"策略
  • 如果列表顺序改变,会导致大量不必要的DOM操作
  • 可能引发状态错乱(如表单元素)

有Key时的优势

  • 精确识别节点身份
  • 最大化复用相同节点
  • 避免不必要的DOM操作

6. 算法优化策略

  1. 首尾指针快速匹配:处理常见的前后添加/删除
  2. Key映射表:O(1)复杂度查找可复用节点
  3. 批量DOM操作:最后统一处理剩余节点的添加/删除
  4. 节点类型判断:不同类型节点直接替换

7. 时间复杂度优化

通过以下策略将O(n³)复杂度优化到O(n):

  • 只比较同层级节点
  • 使用key建立索引
  • 首尾四指针快速跳过相同前缀/后缀

8. 与其他框架的对比

特性VueReact
对比策略双端对比单端递归
Key作用域同一层级内唯一全局唯一
移动节点处理直接移动DOM标记后统一处理
静态节点优化编译时标记不可变数据结构

9. 总结

Vue的Diff算法通过以下方式实现高效更新:

  1. 优先处理常见的前后操作
  2. 利用key实现精确节点匹配
  3. 最小化DOM操作次数
  4. 智能处理节点复用和移动

理解Diff算法的工作原理有助于:

  • 编写更高效的模板代码
  • 合理使用key优化列表渲染
  • 避免不必要的组件重新渲染
  • 深入理解Vue的响应式更新机制

流程图说明补充

  1. 四个指针分别指向新旧子节点的首尾
  2. 优先处理四种可能的匹配情况:
    • 旧头 vs 新头
    • 旧尾 vs 新尾
    • 旧头 vs 新尾
    • 旧尾 vs 新头
  3. 当四种情况都不匹配时,使用key映射表查找
  4. 最后处理剩余的新增/删除节点

通过这种设计,Vue能够在大多数常见操作(如列表项的顺序调整)中达到O(n)的时间复杂度,保证高效的视图更新。
在这里插入图片描述

相关文章:

Vue Diff算法原理深度解析:如何高效更新虚拟DOM?

文章目录 1. 为什么需要Diff算法&#xff1f;2. Diff算法核心原则3. 核心流程图解4. 核心代码实现&#xff08;简化版&#xff09;5. Key的重要性示例6. 算法优化策略7. 时间复杂度优化8. 与其他框架的对比9. 总结 1. 为什么需要Diff算法&#xff1f; 在Vue的响应式系统中&…...

Dify平台部署记录

安装dify项目 官网地址&#xff1a;http://difyai.com/ github地址&#xff1a;https://github.com/langgenius/dify 下载项目&#xff1a; git clone https://github.com/langgenius/dify.git下载过慢&#xff0c;直接访问网页下载zip压缩包&#xff1a; 解压&#xff0c;…...

ArcGIS Pro中字段的新建方法与应用

一、引言 在地理信息系统&#xff08;GIS&#xff09;的数据管理和分析过程中&#xff0c;字段操作起着至关重要的作用。 无论是进行地图制作、空间分析还是数据统计&#xff0c;字段都是承载属性信息的基本单元。 ArcGIS Pro作为一款功能强大的GIS软件&#xff0c;为用户提…...

Git 的基本概念和使用方式。

Git 是一种分布式版本控制系统&#xff0c;用于跟踪文件和目录的变化。Git 的基本概念和使用方式如下&#xff1a; 仓库&#xff08;Repository&#xff09;&#xff1a;Git 仓库是用来存储项目文件和历史记录的地方。一个 Git 仓库包含项目的文件、版本记录和配置信息。 提交…...

贪心算法--

1.柠檬水找零 link:860. 柠檬水找零 - 力扣&#xff08;LeetCode&#xff09; code class Solution { public:bool lemonadeChange(vector<int>& bills) {// 贪心算法&#xff0c; 优先花出大面额bill&#xff0c; 尽可能保护小面额billint five 0, ten 0;// 不…...

mysql下载与安装、关系数据库和表的创建

一、mysql下载&#xff1a; MySQL获取&#xff1a; 官网&#xff1a;www.mysql.com 也可以从Oracle官方进入&#xff1a;https://www.oracle.com/ 下载地址&#xff1a;https://downloads.mysql.com/archives/community/ 选择对应的版本和对应的操作系统&a…...

万字技术指南STM32F103C8T6 + ESP8266-01 连接 OneNet 平台 MQTT/HTTP

此博客为一份详细的指南&#xff0c;涵盖 STM32F103C8T6 通过 ESP8266-01 连接 OneNet 平台&#xff0c;并使用 MQTT/HTTP 进行数据通信的完整流程。这份文档包括&#xff1a; OneNet 平台的介绍与功能概览在 OneNet 上创建和配置设备的方法STM32CubeIDE 的开发环境搭建ESP826…...

MWC 2025 | 紫光展锐联合移远通信推出全面支持R16特性的5G模组RG620UA-EU

2025年世界移动通信大会&#xff08;MWC 2025&#xff09;期间&#xff0c;紫光展锐联合移远通信&#xff0c;正式发布了全面支持5G R16特性的模组RG620UA-EU&#xff0c;以强大的灵活性和便捷性赋能产业。 展锐芯加持&#xff0c;关键性能优异 RG620UA-EU模组基于紫光展锐V62…...

PyCharm 接入 DeepSeek、OpenAI、Gemini、Mistral等大模型完整版教程(通用)!

PyCharm 接入 DeepSeek、OpenAI、Gemini、Mistral等大模型完整版教程&#xff08;通用&#xff09;&#xff01; 当我们成功接入大模型时&#xff0c;可以选中任意代码区域进行解答&#xff0c;共分为三个区域&#xff0c;分别是选中区域、提问区域以及回答区域&#xff0c;我…...

小智智能体语言大模型硬件软件开发

硬件可以参考ESP32-AI语音助手 - 立创开源硬件平台 单片机使用esp32s3&#xff0c;可以直接替换&#xff0c;但是引脚IO有变化&#xff0c;而且esp32s3 io35 36 37不能用&#xff0c;所以得飞一条线&#xff0c;原先接在io35的飞到io4上。如果不飞线的话系统一直重启 软件使用…...

网络tcp协议设置,网络tcp协议设置不了

网络TCP协议的设置通常涉及到多个方面&#xff0c;包括IP地址、子网掩码、默认网关、DNS服务器等参数的配置&#xff0c;以及TCP/IP协议栈本身的配置。如果遇到网络TCP协议设置不了的问题&#xff0c;可能是由多种原因导致的。以下是一些可能的原因及解决方法&#xff1a; 一、…...

配置Hadoop集群

Hadoop的运行模式 本地运行&#xff1a;在一台单机上运行&#xff0c;没有分布式文件系统&#xff0c;直接读写本地操作系统的文件系统。特点&#xff1a;不对配置文件进行修改&#xff0c;Hadoop 不会启动 伪分布式&#xff1a;也是在一台单机上运行&#xff0c;但用不同的 …...

模型微调-基于LLaMA-Factory进行微调的一个简单案例

模型微调-基于LLaMA-Factory进行微调的一个简单案例 1. 租用云计算资源2. 拉取 LLaMa-Factory3. 安装依赖环境4. 启动 LLaMa-Factory 界面5. 从 Huggingface 下载模型6. 模型验证7. 模型微调 1. 租用云计算资源 以下示例基于 AutoDL 云计算资源。 在云计算平台选择可用的云计…...

设置重定向不缓存

response.setHeader(“Cache-Control”, “no-cache, no-store, must-revalidate”); response.setHeader(“Pragma”, “no-cache”);response.setHeader(“Expires”, “0”);response.sendRedirect(newURL); response.setContentType(“text/html;charsetUTF-8”); PrintWr…...

java-算法基础优化

一、ACM风格输入输出&#xff08;高效&#xff0c;替换原有的输入输出流&#xff09; 1.推荐原因&#xff1a;&#xff08;内存托管&#xff09; 对于原本的Scanner读取流&#xff0c;只能根据行来读取数据&#xff0c;而BufferredReader读取信息可以直接读取整个文件&#xf…...

⚡ 回声谷即时通讯系统

基于SpringBootVue3的实时通信解决方案 &#x1f31f; 核心特性 #mermaid-svg-uxEwEcjlUVI6Tjjf {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-uxEwEcjlUVI6Tjjf .error-icon{fill:#552222;}#mermaid-svg-uxEwEcjl…...

《 PyQt5》—— 创建 Python GUI(图形用户界面)

文章目录 PyQt5安装基本概念进行配置配置QtDesigner配置PyUIC配置Pyrcc 使用PyQt5使用如何使用ui文件 PyQt5 PyQt5 是一个用于创建 Python GUI&#xff08;图形用户界面&#xff09;应用程序的强大工具包&#xff0c;它是 Qt 应用程序框架的 Python 绑定。Qt 是一个跨平台的 C…...

Python图形编程之EasyGUI: indexbox的用法

目录<<上一章&#xff1a;ynbox用法详解 下一章&#xff1a;boolbox用法详解 >> # 1 Python图形编程之EasyGUI: indexbox的用法 1.1 基本用法 indexbox提供用户一个选择不同选项的功能&#xff0c;不同的选项由按钮来表示&#xff0c;提供类似功能的还有choicebox…...

vue+dhtmlx-gantt 实现甘特图-快速入门【甘特图】

文章目录 一、前言二、使用说明2.1 引入依赖2.2 引入组件2.3 引入dhtmlx-gantt2.4 甘特图数据配置2.5 初始化配置 三、代码示例3.1 Vue2完整示例3.2 Vue3 完整示例 四、效果图 一、前言 dhtmlxGantt 是一款功能强大的甘特图组件&#xff0c;支持 Vue 3 集成。它提供了丰富的功…...

游戏引擎学习第147天

仓库:https://gitee.com/mrxiao_com/2d_game_3 上一集回顾 具体来说&#xff0c;我们通过隐式计算来解决问题&#xff0c;而不是像数字微分分析器那样逐步增加数据。我们已经涵盖了这个部分&#xff0c;并计划继续处理音量问题。不过&#xff0c;实际上我们现在不需要继续处理…...

Python自动点击器开发教程 - 支持键盘连按和鼠标连点

Python自动点击器开发教程 - 支持键盘连按和鼠标连点 这里写目录标题 Python自动点击器开发教程 - 支持键盘连按和鼠标连点项目介绍开发环境安装依赖核心代码解析1. 键盘模拟实现2. 鼠标点击实现 开发要点使用说明注意事项优化建议打包发布项目源码开发心得参考资料成品工具 项…...

C++ 链表List使用与实现:拷贝交换与高效迭代器细致讲解

目录 list的使用&#xff1a; 构造与赋值 元素访问 修改操作 容量查询 链表特有操作 拼接&#xff08;Splice&#xff09; C11 新增方法 注意&#xff1a; stl_list的模拟实现&#xff1a; 一、链表节点设计的艺术 1.1 结构体 vs 类的选择 二、迭代器实现的精髓 2…...

Manus联创澄清:我们并未使用MCP技术

摘要 近日&#xff0c;Manus联创针对外界关于其产品可能涉及“沙盒越狱”的疑问进行了正式回应。公司明确表示并未使用Anthropic的MCP&#xff08;模型上下文协议&#xff09;技术&#xff0c;并强调MCP是一个旨在标准化应用程序与大型语言模型&#xff08;LLM&#xff09;之间…...

ACE学习2——write transaction

用于处理缓存行的数据更新到主内存&#xff08;main memory&#xff09;的操作。 以下是用于更新主内存的几种事务类型&#xff1a; WriteBack&#xff1a; WriteBack事务用于将cache中的dirty态的cacheline写回主存&#xff0c;以释放cache中的cacheline&#xff0c;用于存…...

c++ 返回引用

在C中&#xff0c;返回引用是一种常见的做法&#xff0c;特别是在需要返回大型对象时&#xff0c;以避免不必要的复制&#xff0c;从而提高程序的效率。返回引用通常有两种情况&#xff1a;返回局部变量的引用和返回成员变量的引用。下面分别讨论这两种情况以及如何安全地实现它…...

Docker篇

1.docker环境搭建&#xff1a; 1.1软件仓库的配置rhel9&#xff1a; #cd/etc/yum.repos.d #vim docker.repo [docker] namedocker-ce baseurlhttps://mirrors.aliyun.com/docker-ce/linux/rhel/9/x86_64/stable gpgcheck0 1.2安装docker并且启动服务 yum install -y dock…...

TypeScript基础类型详解:与JavaScript的对比与核心价值

TypeScript作为JavaScript的超集&#xff0c;最大的特性是引入了静态类型系统。本文将基于TypeScript官网内容&#xff0c;解析其基础类型设计&#xff0c;并与ES/JavaScript进行对比&#xff0c;揭示类型系统的实际价值。 一、基础类型全景图 1. 原生类型的强化 JavaScript原…...

Linux《基础开发工具(中)》

在之前的Linux《基础开发工具&#xff08;上&#xff09;》当中已经了解了Linux当中到的两大基础的开发工具yum与vim&#xff1b;了解了在Linux当中如何进行软件的下载以及实现的基本原理、知道了编辑器vim的基本使用方式&#xff0c;那么接下来在本篇当中将接下去继续来了解另…...

CPU 负载 和 CPU利用率 的区别

简单记录下 top 命令中&#xff0c;CPU利用率核CPU负载的概念&#xff0c; &#xff08;1&#xff09;CPU利用率&#xff1a;指在一段时间内 表示 CPU 实际工作时间占总时间的百分比。表示正在执行进程的时间比例&#xff0c;包括用户空间和内核空间程序的执行时间。通常包含以…...

vue源码(二)

文章目录 数据代理示例 初始化组件实例计算属性基本用法ComputedReflmpl类计算属性的创建 Vue3的特点及优势声明式框架采用虚拟DOM区分编译时和进行时 Vue3设计思想 数据代理 示例 以下代码主要是有一个msg的响应式数据&#xff0c;点击按钮后修改msg的内容。根据代码可知有两…...