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

为OpenClaw智能体工作流配置Taotoken作为统一的模型调用后端

&#x1f680; 告别海外账号与网络限制&#xff01;稳定直连全球优质大模型&#xff0c;限时半价接入中。 &#x1f449; 点击领取海量免费额度 为OpenClaw智能体工作流配置Taotoken作为统一的模型调用后端 对于使用OpenClaw框架构建AI智能体的开发者而言&#xff0c;一个稳定…...

物理网卡down了?虚拟机还能通信吗?看teaming策略就够了

在ESXi虚拟化运维中&#xff0c;物理网卡&#xff08;vmnic&#xff09;故障、网线松动、网卡损坏导致网卡down&#xff08;宕机&#xff09;&#xff0c;是常见的硬件故障场景。很多新手遇到这种情况&#xff0c;会下意识认为所有虚拟机都会断网&#xff0c;但实际并非如此。核…...

从需求到开发的全流程

一、流程图二、各阶段拆解&#x1f50d;第一阶段&#xff1a;需求细化与设计&#xff08;会前关键&#xff09;此阶段的目标是产出一份清晰、可评审的PRD初稿。步骤核心动作与目的产出物与实战技巧1. 深度需求调研目的&#xff1a;消化方案&#xff0c;与原始需求方及关键用户深…...

【C++ 多态】虚函数 · 虚表 · 重写,一篇彻底弄明白!

C 多态详解 C多态是面向对象的核心灵魂&#xff0c;本文将由浅入深&#xff0c;带你循序渐进地掌握多态的方方面面&#xff0c;全程干货&#xff0c;坐稳发车~ ദ്ദി˶&#xff70;̀֊&#xff70;́ )✧ 文章目录C 多态详解1. 什么是多态&#xff1f;2. 运行时多态的实现前…...

如何免费获得Windows风扇智能控制:FanControl终极指南

如何免费获得Windows风扇智能控制&#xff1a;FanControl终极指南 【免费下载链接】FanControl.Releases This is the release repository for Fan Control, a highly customizable fan controlling software for Windows. 项目地址: https://gitcode.com/GitHub_Trending/fa…...

2026年最新英语单词AI辅助工具 帮英语学习者轻松提升背词效率

英语单词学习的核心痛点拆解我们团队做英语学习工具测评快5年了&#xff0c;后台收到最多的提问就是「有没有能真的提升背词效率的工具」&#xff0c;拆解下来行业的共性痛点其实很明确&#xff1a;第一是资源错配&#xff0c;80%的背词时间都花在已经掌握的词汇上&#xff0c;…...

摄像头驱动调试避坑指南:用示波器快速定位I2C不通、MIPI无信号问题

摄像头驱动调试避坑指南&#xff1a;用示波器快速定位I2C不通、MIPI无信号问题 当摄像头模组在硬件调试阶段出现异常时&#xff0c;软件工程师往往会陷入"配置检查-重新烧录-再检查"的死循环。实际上&#xff0c;80%的摄像头初始化失败问题源于硬件信号层面的异常。本…...

告别盲选!深入解读5G NR中UCI偏置值(beta_offset)的配置策略与索引选择

5G NR中UCI偏置值配置的工程实践指南 在5G新空口(NR)系统中&#xff0c;上行控制信息(UCI)通过物理上行共享信道(PUSCH)传输时&#xff0c;其资源分配直接影响到系统性能和用户体验。作为网络优化工程师&#xff0c;我们经常需要面对各种复杂的配置场景&#xff0c;而UCI偏置值…...

从AlphaGo到你的小游戏:如何用MCTS(蒙特卡洛树搜索)为你的五子棋项目加个‘智能大脑’

从AlphaGo到你的小游戏&#xff1a;如何用MCTS为五子棋项目构建智能决策引擎 当你在手机上下棋输给AI时&#xff0c;是否好奇过这些"电子大脑"如何思考&#xff1f;2016年AlphaGo击败李世石的关键技术之一——蒙特卡洛树搜索&#xff08;MCTS&#xff09;&#xff0c…...

青年教师评副高‘捷径’:这6本被低估的SSCI,认可度不输顶刊!

01 Academic Medicine期刊分区影响因子自引率年文章数教育学1区5.211.5%252篇投稿参考&#xff1a;美国医学院协会&#xff08;AAMC&#xff09;官方期刊&#xff0c;审稿周期 2–3 个月&#xff0c;录用率≈20%&#xff1b;可选非 OA 模式免版面费&#xff0c;适合具有实践转…...