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. 核心流程图解
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. 算法优化策略
- 首尾指针快速匹配:处理常见的前后添加/删除
- Key映射表:O(1)复杂度查找可复用节点
- 批量DOM操作:最后统一处理剩余节点的添加/删除
- 节点类型判断:不同类型节点直接替换
7. 时间复杂度优化
通过以下策略将O(n³)复杂度优化到O(n):
- 只比较同层级节点
- 使用key建立索引
- 首尾四指针快速跳过相同前缀/后缀
8. 与其他框架的对比
| 特性 | Vue | React |
|---|---|---|
| 对比策略 | 双端对比 | 单端递归 |
| Key作用域 | 同一层级内唯一 | 全局唯一 |
| 移动节点处理 | 直接移动DOM | 标记后统一处理 |
| 静态节点优化 | 编译时标记 | 不可变数据结构 |
9. 总结
Vue的Diff算法通过以下方式实现高效更新:
- 优先处理常见的前后操作
- 利用key实现精确节点匹配
- 最小化DOM操作次数
- 智能处理节点复用和移动
理解Diff算法的工作原理有助于:
- 编写更高效的模板代码
- 合理使用key优化列表渲染
- 避免不必要的组件重新渲染
- 深入理解Vue的响应式更新机制
流程图说明补充:
- 四个指针分别指向新旧子节点的首尾
- 优先处理四种可能的匹配情况:
- 旧头 vs 新头
- 旧尾 vs 新尾
- 旧头 vs 新尾
- 旧尾 vs 新头
- 当四种情况都不匹配时,使用key映射表查找
- 最后处理剩余的新增/删除节点
通过这种设计,Vue能够在大多数常见操作(如列表项的顺序调整)中达到O(n)的时间复杂度,保证高效的视图更新。

相关文章:
Vue Diff算法原理深度解析:如何高效更新虚拟DOM?
文章目录 1. 为什么需要Diff算法?2. Diff算法核心原则3. 核心流程图解4. 核心代码实现(简化版)5. Key的重要性示例6. 算法优化策略7. 时间复杂度优化8. 与其他框架的对比9. 总结 1. 为什么需要Diff算法? 在Vue的响应式系统中&…...
Dify平台部署记录
安装dify项目 官网地址:http://difyai.com/ github地址:https://github.com/langgenius/dify 下载项目: git clone https://github.com/langgenius/dify.git下载过慢,直接访问网页下载zip压缩包: 解压,…...
ArcGIS Pro中字段的新建方法与应用
一、引言 在地理信息系统(GIS)的数据管理和分析过程中,字段操作起着至关重要的作用。 无论是进行地图制作、空间分析还是数据统计,字段都是承载属性信息的基本单元。 ArcGIS Pro作为一款功能强大的GIS软件,为用户提…...
Git 的基本概念和使用方式。
Git 是一种分布式版本控制系统,用于跟踪文件和目录的变化。Git 的基本概念和使用方式如下: 仓库(Repository):Git 仓库是用来存储项目文件和历史记录的地方。一个 Git 仓库包含项目的文件、版本记录和配置信息。 提交…...
贪心算法--
1.柠檬水找零 link:860. 柠檬水找零 - 力扣(LeetCode) code class Solution { public:bool lemonadeChange(vector<int>& bills) {// 贪心算法, 优先花出大面额bill, 尽可能保护小面额billint five 0, ten 0;// 不…...
mysql下载与安装、关系数据库和表的创建
一、mysql下载: MySQL获取: 官网:www.mysql.com 也可以从Oracle官方进入:https://www.oracle.com/ 下载地址:https://downloads.mysql.com/archives/community/ 选择对应的版本和对应的操作系统&a…...
万字技术指南STM32F103C8T6 + ESP8266-01 连接 OneNet 平台 MQTT/HTTP
此博客为一份详细的指南,涵盖 STM32F103C8T6 通过 ESP8266-01 连接 OneNet 平台,并使用 MQTT/HTTP 进行数据通信的完整流程。这份文档包括: OneNet 平台的介绍与功能概览在 OneNet 上创建和配置设备的方法STM32CubeIDE 的开发环境搭建ESP826…...
MWC 2025 | 紫光展锐联合移远通信推出全面支持R16特性的5G模组RG620UA-EU
2025年世界移动通信大会(MWC 2025)期间,紫光展锐联合移远通信,正式发布了全面支持5G R16特性的模组RG620UA-EU,以强大的灵活性和便捷性赋能产业。 展锐芯加持,关键性能优异 RG620UA-EU模组基于紫光展锐V62…...
PyCharm 接入 DeepSeek、OpenAI、Gemini、Mistral等大模型完整版教程(通用)!
PyCharm 接入 DeepSeek、OpenAI、Gemini、Mistral等大模型完整版教程(通用)! 当我们成功接入大模型时,可以选中任意代码区域进行解答,共分为三个区域,分别是选中区域、提问区域以及回答区域,我…...
小智智能体语言大模型硬件软件开发
硬件可以参考ESP32-AI语音助手 - 立创开源硬件平台 单片机使用esp32s3,可以直接替换,但是引脚IO有变化,而且esp32s3 io35 36 37不能用,所以得飞一条线,原先接在io35的飞到io4上。如果不飞线的话系统一直重启 软件使用…...
网络tcp协议设置,网络tcp协议设置不了
网络TCP协议的设置通常涉及到多个方面,包括IP地址、子网掩码、默认网关、DNS服务器等参数的配置,以及TCP/IP协议栈本身的配置。如果遇到网络TCP协议设置不了的问题,可能是由多种原因导致的。以下是一些可能的原因及解决方法: 一、…...
配置Hadoop集群
Hadoop的运行模式 本地运行:在一台单机上运行,没有分布式文件系统,直接读写本地操作系统的文件系统。特点:不对配置文件进行修改,Hadoop 不会启动 伪分布式:也是在一台单机上运行,但用不同的 …...
模型微调-基于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风格输入输出(高效,替换原有的输入输出流) 1.推荐原因:(内存托管) 对于原本的Scanner读取流,只能根据行来读取数据,而BufferredReader读取信息可以直接读取整个文件…...
⚡ 回声谷即时通讯系统
基于SpringBootVue3的实时通信解决方案 🌟 核心特性 #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(图形用户界面)应用程序的强大工具包,它是 Qt 应用程序框架的 Python 绑定。Qt 是一个跨平台的 C…...
Python图形编程之EasyGUI: indexbox的用法
目录<<上一章:ynbox用法详解 下一章:boolbox用法详解 >> # 1 Python图形编程之EasyGUI: indexbox的用法 1.1 基本用法 indexbox提供用户一个选择不同选项的功能,不同的选项由按钮来表示,提供类似功能的还有choicebox…...
vue+dhtmlx-gantt 实现甘特图-快速入门【甘特图】
文章目录 一、前言二、使用说明2.1 引入依赖2.2 引入组件2.3 引入dhtmlx-gantt2.4 甘特图数据配置2.5 初始化配置 三、代码示例3.1 Vue2完整示例3.2 Vue3 完整示例 四、效果图 一、前言 dhtmlxGantt 是一款功能强大的甘特图组件,支持 Vue 3 集成。它提供了丰富的功…...
游戏引擎学习第147天
仓库:https://gitee.com/mrxiao_com/2d_game_3 上一集回顾 具体来说,我们通过隐式计算来解决问题,而不是像数字微分分析器那样逐步增加数据。我们已经涵盖了这个部分,并计划继续处理音量问题。不过,实际上我们现在不需要继续处理…...
Python爬虫实战:研究MechanicalSoup库相关技术
一、MechanicalSoup 库概述 1.1 库简介 MechanicalSoup 是一个 Python 库,专为自动化交互网站而设计。它结合了 requests 的 HTTP 请求能力和 BeautifulSoup 的 HTML 解析能力,提供了直观的 API,让我们可以像人类用户一样浏览网页、填写表单和提交请求。 1.2 主要功能特点…...
生成xcframework
打包 XCFramework 的方法 XCFramework 是苹果推出的一种多平台二进制分发格式,可以包含多个架构和平台的代码。打包 XCFramework 通常用于分发库或框架。 使用 Xcode 命令行工具打包 通过 xcodebuild 命令可以打包 XCFramework。确保项目已经配置好需要支持的平台…...
Android Wi-Fi 连接失败日志分析
1. Android wifi 关键日志总结 (1) Wi-Fi 断开 (CTRL-EVENT-DISCONNECTED reason3) 日志相关部分: 06-05 10:48:40.987 943 943 I wpa_supplicant: wlan0: CTRL-EVENT-DISCONNECTED bssid44:9b:c1:57:a8:90 reason3 locally_generated1解析: CTR…...
React hook之useRef
React useRef 详解 useRef 是 React 提供的一个 Hook,用于在函数组件中创建可变的引用对象。它在 React 开发中有多种重要用途,下面我将全面详细地介绍它的特性和用法。 基本概念 1. 创建 ref const refContainer useRef(initialValue);initialValu…...
Java如何权衡是使用无序的数组还是有序的数组
在 Java 中,选择有序数组还是无序数组取决于具体场景的性能需求与操作特点。以下是关键权衡因素及决策指南: ⚖️ 核心权衡维度 维度有序数组无序数组查询性能二分查找 O(log n) ✅线性扫描 O(n) ❌插入/删除需移位维护顺序 O(n) ❌直接操作尾部 O(1) ✅内存开销与无序数组相…...
全球首个30米分辨率湿地数据集(2000—2022)
数据简介 今天我们分享的数据是全球30米分辨率湿地数据集,包含8种湿地亚类,该数据以0.5X0.5的瓦片存储,我们整理了所有属于中国的瓦片名称与其对应省份,方便大家研究使用。 该数据集作为全球首个30米分辨率、覆盖2000–2022年时间…...
ffmpeg(四):滤镜命令
FFmpeg 的滤镜命令是用于音视频处理中的强大工具,可以完成剪裁、缩放、加水印、调色、合成、旋转、模糊、叠加字幕等复杂的操作。其核心语法格式一般如下: ffmpeg -i input.mp4 -vf "滤镜参数" output.mp4或者带音频滤镜: ffmpeg…...
微服务商城-商品微服务
数据表 CREATE TABLE product (id bigint(20) UNSIGNED NOT NULL AUTO_INCREMENT COMMENT 商品id,cateid smallint(6) UNSIGNED NOT NULL DEFAULT 0 COMMENT 类别Id,name varchar(100) NOT NULL DEFAULT COMMENT 商品名称,subtitle varchar(200) NOT NULL DEFAULT COMMENT 商…...
ip子接口配置及删除
配置永久生效的子接口,2个IP 都可以登录你这一台服务器。重启不失效。 永久的 [应用] vi /etc/sysconfig/network-scripts/ifcfg-eth0修改文件内内容 TYPE"Ethernet" BOOTPROTO"none" NAME"eth0" DEVICE"eth0" ONBOOT&q…...
Pinocchio 库详解及其在足式机器人上的应用
Pinocchio 库详解及其在足式机器人上的应用 Pinocchio (Pinocchio is not only a nose) 是一个开源的 C 库,专门用于快速计算机器人模型的正向运动学、逆向运动学、雅可比矩阵、动力学和动力学导数。它主要关注效率和准确性,并提供了一个通用的框架&…...
