当前位置: 首页 > 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;实际上我们现在不需要继续处理…...

第19节 Node.js Express 框架

Express 是一个为Node.js设计的web开发框架&#xff0c;它基于nodejs平台。 Express 简介 Express是一个简洁而灵活的node.js Web应用框架, 提供了一系列强大特性帮助你创建各种Web应用&#xff0c;和丰富的HTTP工具。 使用Express可以快速地搭建一个完整功能的网站。 Expre…...

Redis相关知识总结(缓存雪崩,缓存穿透,缓存击穿,Redis实现分布式锁,如何保持数据库和缓存一致)

文章目录 1.什么是Redis&#xff1f;2.为什么要使用redis作为mysql的缓存&#xff1f;3.什么是缓存雪崩、缓存穿透、缓存击穿&#xff1f;3.1缓存雪崩3.1.1 大量缓存同时过期3.1.2 Redis宕机 3.2 缓存击穿3.3 缓存穿透3.4 总结 4. 数据库和缓存如何保持一致性5. Redis实现分布式…...

MFC内存泄露

1、泄露代码示例 void X::SetApplicationBtn() {CMFCRibbonApplicationButton* pBtn GetApplicationButton();// 获取 Ribbon Bar 指针// 创建自定义按钮CCustomRibbonAppButton* pCustomButton new CCustomRibbonAppButton();pCustomButton->SetImage(IDB_BITMAP_Jdp26)…...

Neo4j 集群管理:原理、技术与最佳实践深度解析

Neo4j 的集群技术是其企业级高可用性、可扩展性和容错能力的核心。通过深入分析官方文档,本文将系统阐述其集群管理的核心原理、关键技术、实用技巧和行业最佳实践。 Neo4j 的 Causal Clustering 架构提供了一个强大而灵活的基石,用于构建高可用、可扩展且一致的图数据库服务…...

从零实现STL哈希容器:unordered_map/unordered_set封装详解

本篇文章是对C学习的STL哈希容器自主实现部分的学习分享 希望也能为你带来些帮助~ 那咱们废话不多说&#xff0c;直接开始吧&#xff01; 一、源码结构分析 1. SGISTL30实现剖析 // hash_set核心结构 template <class Value, class HashFcn, ...> class hash_set {ty…...

Spring Boot+Neo4j知识图谱实战:3步搭建智能关系网络!

一、引言 在数据驱动的背景下&#xff0c;知识图谱凭借其高效的信息组织能力&#xff0c;正逐步成为各行业应用的关键技术。本文聚焦 Spring Boot与Neo4j图数据库的技术结合&#xff0c;探讨知识图谱开发的实现细节&#xff0c;帮助读者掌握该技术栈在实际项目中的落地方法。 …...

3403. 从盒子中找出字典序最大的字符串 I

3403. 从盒子中找出字典序最大的字符串 I 题目链接&#xff1a;3403. 从盒子中找出字典序最大的字符串 I 代码如下&#xff1a; class Solution { public:string answerString(string word, int numFriends) {if (numFriends 1) {return word;}string res;for (int i 0;i &…...

微软PowerBI考试 PL300-在 Power BI 中清理、转换和加载数据

微软PowerBI考试 PL300-在 Power BI 中清理、转换和加载数据 Power Query 具有大量专门帮助您清理和准备数据以供分析的功能。 您将了解如何简化复杂模型、更改数据类型、重命名对象和透视数据。 您还将了解如何分析列&#xff0c;以便知晓哪些列包含有价值的数据&#xff0c;…...

高效线程安全的单例模式:Python 中的懒加载与自定义初始化参数

高效线程安全的单例模式:Python 中的懒加载与自定义初始化参数 在软件开发中,单例模式(Singleton Pattern)是一种常见的设计模式,确保一个类仅有一个实例,并提供一个全局访问点。在多线程环境下,实现单例模式时需要注意线程安全问题,以防止多个线程同时创建实例,导致…...

GruntJS-前端自动化任务运行器从入门到实战

Grunt 完全指南&#xff1a;从入门到实战 一、Grunt 是什么&#xff1f; Grunt是一个基于 Node.js 的前端自动化任务运行器&#xff0c;主要用于自动化执行项目开发中重复性高的任务&#xff0c;例如文件压缩、代码编译、语法检查、单元测试、文件合并等。通过配置简洁的任务…...