Loading [MathJax]/jax/output/CommonHTML/config.js
前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >常见框架的 Diff 算法

常见框架的 Diff 算法

原创
作者头像
HZFEStudio
修改于 2021-11-01 01:38:56
修改于 2021-11-01 01:38:56
8590
举报
文章被收录于专栏:HZFEStudioHZFEStudio

完整高频题库仓库地址:https://github.com/hzfe/awesome-interview

完整高频题库阅读地址:https://febook.hzfe.org/

相关问题

  • 虚拟 DOM 是什么
  • 虚拟 DOM 的作用
  • 讲一下 Vue 的 Diff 算法

回答关键点

虚拟 DOM 时间复杂度O(n)

现代网站大多具有复杂布局,大量的节点和交互操作等特征,直接操作 DOM 方法不当带来的性能问题不可忽视。虚拟 DOM 的本质是 JavaScript 对象,它可以代表 DOM 的一部分特征,是 DOM 的抽象简化版本。通过预先操作虚拟 DOM,在某个时机找出和真实 DOM 之间的差异部分并重新渲染,来提升操作真实 DOM 的性能和效率。

为达到这个目的,还需要关注两个问题:什么时候重新渲染,怎么高效选择重新渲染的范围。找出需要重新渲染的范围,就是 Diff 的过程。React 和 Vue 的 Diff 算法思路基本一致,只对同层节点进行比较,利用唯一标识符对节点进行区分。

知识点深入

1. Diff 算法

两棵树的比对和更新,涉及到树编辑距离(Tree Editing Distance)算法:将一棵树转化为另一棵树的最小操作成本。操作类型包括:删除、插入、修改。时间复杂度为 O(n^3)。

为了降低时间复杂度,React 和 Vue 的思路是基于以下两个假设条件,缩减递归迭代规模,将 Diff 算法的时间复杂度降低为 O(n):

  1. 相同类型的组件产生相同的 DOM 结构,反之亦然。所以不同类型组件的结构不需要进一步递归 Diff。
  2. 同一层级的一组节点,可以通过唯一标识符进行区分。

2. React Reconciliation

在 React 中,将虚拟 DOM 和真实 DOM 进行比对然后同步的过程被称为 Reconciliation(调和),Fiber 是 React 16 中新的调和引擎。它的主要目标是实现虚拟 DOM 的增量渲染。

Diff 的大致过程是,当对比两棵虚拟 DOM 树时,React 先对比根元素。依据根元素的类型不同,会有不同的操作:

  • 不同类型的元素 如果元素的类型不同,React 会抛弃旧树并建立新树。如以下情况,会导致完全重建:
代码语言:txt
AI代码解释
复制
<!-- old -->
<button class="bg-blue-100">HZFE</button>

<!-- new -->
<div class="bg-blue-100">HZFE</div>
  • 相同类型的元素 如果元素是两个相同类型的 React DOM 元素时,React 会查看两者的属性,保留 DOM 节点,只更新改变的属性。如以下情况,React 只更新颜色样式。
代码语言:txt
AI代码解释
复制
<!-- old -->
<button class="bg-blue-100 text-center">HZFE</button>

<!-- new -->
<button class="bg-red-100 text-center">HZFE</button>

在元素类型相同的情况下,比对完元素后,会递归元素的子元素。默认情况下,React 会同时迭代新老两个子元素列表。对于列表的更新,React 建议在列表项中标识 key 属性。避免以下低效场景:

代码语言:txt
AI代码解释
复制
<!-- bad -->
<!-- React 不会意识到可以保留<li>HZFE</li>和<li>Front-End</li>子树的完整,而是重写每个元素 -->

<!-- old -->
<ul>
  <li>HZFE</li>
  <li>Front-End</li>
</ul>
<!-- new -->
<ul>
  <li>Back-End</li>
  <li>HZFE</li>
  <li>Front-End</li>
</ul>

<!-- good -->
<!-- 子列表项有稳定且在兄弟节点中唯一的 key 属性, -->
<!-- React 使用 key 从新老树中匹配对应节点比较,提高 Diff 效率。 -->

<!-- old -->
<ul>
  <li key="2016">HZFE</li>
  <li key="2017">Front-End</li>
</ul>
<!-- new -->
<ul>
  <li key="2015">Back-End</li>
  <li key="2016">HZFE</li>
  <li key="2017">Front-End</li>
</ul>

2. Vue2.x Diff

Vue 的 Diff 算法和 React 的类似,只在同一层次进行比较,不进行跨层比较。如果两个元素被判定为不相同,则不继续递归比较。在 Diff 子元素的过程中,采用双端比较的方法,设立 4 个指针:

  • oldStartIdx 指向旧子元素列表中,从左边开始 Diff 的元素索引。初始值:第一个元素的索引。
  • newStartIdx 指向新子元素列表中,从左边开始 Diff 的元素索引。初始值:第一个元素的索引。
  • oldEndIdx 指向旧子元素列表中,从右边开始 Diff 的元素索引。初始值:最后一个元素的索引。
  • newEndIdx 指向新子元素列表中,从右边开始 Diff 的元素索引。初始值:最后一个元素的索引。
image.png
image.png

Vue 同时遍历新老子元素虚拟 DOM 列表,并采用头尾比较。一般有 4 种情况:

  1. 当新老 start 指针指向的是相同节点 复用节点并按需更新。 新老 start 指针向右移动一位。
  2. 当新老 end 指针指向的是相同节点 复用节点并按需更新。 新老 end 指针向左移动一位。
  3. 当老 start 指针和新 end 指针指向的是相同节点 复用节点并按需更新,将节点对应的真实 DOM 移动到子元素列表队尾。 老 start 指针向右移动一位。 新 end 指针向左移动一位。
  4. 当老 end 指针和新 start 指针指向的是相同节点 复用节点并按需更新,将节点对应的真实 DOM 移动到子元素列表队头。 老 end 指针向左移动一位。 新 start 指针向右移动一位。

在不满足以上情况的前提下,会尝试检查新 start 指针指向的节点是否有唯一标识符 key,如果有且能在旧列表中找到拥有相同 key 的相同类型节点,则可复用并按需更新,且移动节点到新的位置。新 start 指针向右移动一位。如果依旧不满足条件,则新增相关节点。

当新老列表的中任意一个列表的头指针索引大于尾指针索引时,循环遍历结束,按需删除或新增相关节点即可。

参考资料

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
理解DOM Diff算法
虚拟 DOM 出现的背景:在 jQuery 时代,可以自行控制 DOM 操作的时机,手动调整,但是当项目很大时,操作 DOM 的复杂度就会上来,DOM 操作会很耗费性能,操作 DOM 就还需要考虑优化 DOM 操作,提升性能。《高性能 JavaScript》这本书中说,把 DOM 和 JavaScript 各自想象成一个岛屿,它们之间用收费桥梁连接。操作 DOM 后需要经过跨流程通信和渲染线程触发的重新渲染(重绘或者重排),在开发中,应尽量减少操作 DOM。而虚拟 DOM 出现后,更新 DOM 交给框架处理。操作虚拟 DOM 可能并没有操作真实 DOM 快,但是它让开发人员不再把很多精力放在操作 DOM 上,而是专注于处理业务数据。本文以 Vue 原码中的 DOM diff 算法为例,介绍一下这个算法的实现原理。
多云转晴
2020/07/29
1.1K0
理解DOM Diff算法
React中diff算法的理解
diff算法用来计算出Virtual DOM中改变的部分,然后针对该部分进行DOM操作,而不用重新渲染整个页面,渲染整个DOM结构的过程中开销是很大的,需要浏览器对DOM结构进行重绘与回流,而diff算法能够使得操作过程中只更新修改的那部分DOM结构而不更新整个DOM,这样能够最小化操作DOM结构,能够最大程度上减少浏览器重绘与回流的规模。
WindRunnerMax
2021/05/20
1.2K0
React的diffing算法学习
这段时间主要在学习React的使用,React是一个用于构建用户界面的框架,其使用了一些方式来使得视图渲染更加高效,这里主要记录一下学习期间了解到的Diffing方法相关的内容。
IMWeb前端团队
2019/12/03
6600
彻底搞懂Vue虚拟Dom和diff算法
使用过Vue和React的小伙伴肯定对虚拟Dom和diff算法很熟悉,它扮演着很重要的角色。由于小编接触Vue比较多,React只是浅学,所以本篇主要针对Vue来展开介绍,带你一步一步搞懂它。
yyds2026
2022/10/17
8240
react学习(八) diff 算法实现
前面几节我们学习了解了 react 的渲染机制和生命周期,本节我们正式进入基本面试必考的核心地带 -- diff 算法,了解如何优化和复用 dom 操作的,还有我们常见的 key 的作用。
测不准
2022/05/22
1K0
react学习(八) diff 算法实现
老生常谈React的diff算法原理-面试版
从代码可以看出,React通过先判断key是否相同,如果key相同则判断type是否相同,只有都相同时一个DOM节点才能复用。
beifeng1996
2022/10/05
5810
React面试:谈谈虚拟DOM,Diff算法与Key机制
原生的JS DOM操作非常消耗性能,而React把真实原生JS DOM转换成了JavaScript对象。这就是虚拟Dom(Virtual Dom)
beifeng1996
2022/10/01
1.5K0
重谈react优势——react技术栈回顾
现在,react已经慢慢退火,该用用react技术栈的已经使用上,填过多少坑,加过多少班,血泪控诉也不下千文。
周陆军
2018/08/03
1.3K0
Fiber:React 的性能保障
React 的更新分为两大阶段,分别是 Reconciliation 阶段和 Commit 阶段。
奋飛
2024/05/25
1570
Fiber:React 的性能保障
为什么 Vue 中不要用 index 作为 key?(diff 算法详解)
Vue 中的 key 是用来做什么的?为什么不推荐使用 index 作为 key?常常听说这样的问题,本篇文章带你从原理来一探究竟。
ssh_晨曦时梦见兮
2020/04/11
9340
为什么 React 的 Diff 算法不采用 Vue 的双端对比算法?
都说“双端对比算法”,那么双端对比算法,到底是怎么样的呢?跟 React 中的 Diff 算法又有什么不同呢?
科技新语
2022/07/06
8170
为什么 React 的 Diff 算法不采用 Vue 的双端对比算法?
前端基础知识整理汇总(下)
接上前面两期的内容,《前端基础知识整理汇总(上)》、《前端基础知识整理汇总(中)》,如果你还没有看前面内容的话,建议你可以点开连接看看,也可以收藏着有空的时候,慢慢看。
前端达人
2021/03/16
1.1K0
前端基础知识整理汇总(下)
vue面试常见考察点总结
beforeCreate 在实例初始化之后,数据观测(data observer) 和 event/watcher 事件配置之前被调用。在当前阶段 data、methods、computed 以及 watch 上的数据和方法都不能被访问
bb_xiaxia1998
2022/10/13
8770
深入 Vue2.x 的虚拟 DOM diff 原理
serena
2017/09/27
7.9K0
深入 Vue2.x 的虚拟 DOM diff 原理
「源码剖析」如何实现一个虚拟DOM算法
上篇文章《虚拟DOM如何进化为真实DOM》中讲到了如何通过虚拟DOM树转化为真实DOM渲染到页面中。但是在渲染的过程中,我们直接将新的虚拟DOM树转化成真实DOM替换掉旧的DOM结构。当真实的DOM中的状态或者内容发生变化的时候,重新渲染新的虚拟DOM树再替换掉旧的,这样的话会显得很无力。
小丑同学
2021/02/07
6780
Vue中diff算法的理解
diff算法用来计算出Virtual DOM中改变的部分,然后针对该部分进行DOM操作,而不用重新渲染整个页面,渲染整个DOM结构的过程中开销是很大的,需要浏览器对DOM结构进行重绘与回流,而diff算法能够使得操作过程中只更新修改的那部分DOM结构而不更新整个DOM,这样能够最小化操作DOM结构,能够最大程度上减少浏览器重绘与回流的规模。
WindRunnerMax
2020/08/27
7250
字节前端一面常见vue面试题(必备)_2023-02-28
Vue 的编译过程就是将 template 转化为 render 函数的过程 分为以下三步
用户10377014
2023/02/28
6320
探索 React 内核:深入 Fiber 架构和协调算法
深入研究 React 称为 Fiber 的新架构,了解新 reconciliation 算法的两个主要阶段。
童欧巴
2020/07/17
2.3K0
探索 React 内核:深入 Fiber 架构和协调算法
请阐述vue的diff算法
diff是什么?diff就是比较两棵树,render会生成两颗树,一棵新树newVnode,一棵旧树oldVnode,然后两棵树进行对比更新找差异就是diff,全称difference,在vue里面 diff 算法是通过patch函数来完成的,所以有的时候也叫patch算法
程序员法医
2021/09/10
5630
Vue进阶 Diff算法详解
虚拟DOM就是把真实DOM树的结构和信息抽象出来,以对象的形式模拟树形结构,如下:
前端LeBron
2021/12/08
6380
Vue进阶 Diff算法详解
相关推荐
理解DOM Diff算法
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档