Vue diff算法

2024-06-17 09:44

本文主要是介绍Vue diff算法,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

vue diff算法主要体现在renderer.ts中的patchChildren这段代码逻辑。差异算法排序分为无key时的diff算法排序逻辑和有key时的diff算法排序逻辑。无key时的逻辑主要有三步,首先,通过for循环patch重新渲染元素进行替换,其次是删除旧的元素,再次是新增元素。
代码如下:

const patchUnkeyedChildren = ( c1: VNode[], c2: VNodeArrayChildren, container: RendererElement, anchor: RendererNode | null, parentComponent: ComponentInternalInstance | null, parentSuspense: SuspenseBoundary | null, isSVG: boolean, slotScopeIds: string[], optimized: boolean ) => {c1 = c1 || EMPTY_ARR c2 = c2 || EMPTY_ARR const oldLength = c1.length const newLength = c2.length const commonLength = Math.min(oldLength, newLength) for (let i = 0; i < commonLength; i++) { const nextChild = (c2[i] = optimized ? cloneIfMounted(c2[i] as VNode) : normalizeVNode(c2[i])) patch( c1[i], nextChild, container, null , parentComponent, parentSuspense, isSVG, slotScopeIds, optimized ) } if (oldLength > newLength) { unmountChildren( c1, parentComponent, parentSuspense, true , false , commonLength ) } else { mountChildren( c2, container, anchor, parentComponent, parentSuspense, isSVG, slotScopeIds, optimized, commonLength ) } } 
  • 当节点有key时,排序逻辑如下:
    1. 前序算法: 前面的元素与后面的元素比较,若不同,则跳出循环。
    2. 尾序算法: 尾部与尾部比较,若不同,跳出循环。
    3. 若新节点多出来,则将其挂载(即新增)。
    4. 若旧节点多出来,则将其卸载(即删除)。
    5. 在有key的情况下,乱序(涉及最长递增子序列算法),此过程包括构建新节点的映射关系等步骤。

代码示例:

let j 
let patched = 0 
const toBePatched = e2 - s2 + 1 
let moved = false 
let maxNewIndexSoFar = 0
const newIndexToOldIndexMap = new Array(toBePatched)
for (i = 0; i < toBePatched; i++) newIndexToOldIndexMap[i] = 0

最长递增子序列(Longest Increasing Subsequence)

需要一个数组来存储新序列新的索引到旧的索列旧的索引的映射关系。即,newIndexToOldIndexMap变量,它的初始化方式如下:

const toBePatched = e2 - s2 + 1; // e2和s2分别是新旧节点数组的结束和开始索引
let newIndexToOldIndexMap = new Array(toBePatched).fill(0); //初始化映射表

然后在遍历过程中,如果新的VNode与旧的VNode相同,我们将更新这个映射表,如下所示:

newIndexToOldIndexMap[i - s2] = oldIndex + 1;

接下来,我们需要用到一个辅助函数,用来获取最长递增子序列。我们使用动态规划(DP)和二分搜索(Binary Search)来实现这个功能。

对于最长递增子序列的问题,通常我们会用到动态规划(DP)或二分搜索(Binary Search)的方法。下面的代码是一个常规的DP解决方案:

假设 sequence 是输入的数组,length 是数组的长度,我们维护一个 dp 数组来表示以 i 结尾的最长递增子序列的长度:

  let dp = Array(sequence.length).fill(1);for (let i = 1; i < sequence.length; i++) {for (let j = 0; j < i; j++) {if (sequence[i] > sequence[j]) {dp[i] = Math.max(dp[i], dp[j] + 1);}}}return Math.max(...dp);

然而,这个算法的时间复杂度是 O(n^2),若数组的长度很大,可能会非常耗时。为了提高效率,我们可以使用二分搜索的方式,下面是其代码实现以及相应的注释:

let dp = [], len = sequence.length;
for(let i = 0; i < len; i++) {let low = 0, high = dp.length, mid;while(low < high) { //进行二分查找mid = (low + high) >>> 1;if(dp[mid] < sequence[i]) low = mid + 1;  //若当前值大于中间值,则在后半段查找else high = mid;    //否则在前半段查找}dp[low] = sequence[i]; //更新对应位置的值if(low === dp.length)  //若low等于当前dp数组的长度,说明该值可以拓展当前的递增序列dp.push(sequence[i]);
}
return dp.length; //返回最长递增子序列的长度

上节函数中,dp[i] 存储的是长度为 i+1 的递增子序列中末尾的元素值。若存在多个长度为 i+1 的递增子序列,我们选择末尾值最小的那个进行存储,因为末尾值越小,越有可能在其后面追加其他元素。通过二分搜索,我们保证了 dp[] 数组在遍历的过程中,始终保持递增的状态。

以上就是文章全部内容了,如果喜欢这篇文章的话,还希望三连支持一下,感谢!

这篇关于Vue diff算法的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1069106

相关文章

CSS3 布局样式及其应用举例

《CSS3布局样式及其应用举例》CSS3的布局特性为前端开发者提供了无限可能,无论是Flexbox的一维布局还是Grid的二维布局,它们都能够帮助开发者以更清晰、简洁的方式实现复杂的网页布局,本文给... 目录深入探讨 css3 布局样式及其应用引言一、CSS布局的历史与发展1.1 早期布局的局限性1.2

使用animation.css库快速实现CSS3旋转动画效果

《使用animation.css库快速实现CSS3旋转动画效果》随着Web技术的不断发展,动画效果已经成为了网页设计中不可或缺的一部分,本文将深入探讨animation.css的工作原理,如何使用以及... 目录1. css3动画技术简介2. animation.css库介绍2.1 animation.cs

CSS引入方式和选择符的讲解和运用小结

《CSS引入方式和选择符的讲解和运用小结》CSS即层叠样式表,是一种用于描述网页文档(如HTML或XML)外观和格式的样式表语言,它主要用于将网页内容的呈现(外观)和结构(内容)分离,从而实现... 目录一、前言二、css 是什么三、CSS 引入方式1、行内样式2、内部样式表3、链入外部样式表四、CSS 选

使用雪花算法产生id导致前端精度缺失问题解决方案

《使用雪花算法产生id导致前端精度缺失问题解决方案》雪花算法由Twitter提出,设计目的是生成唯一的、递增的ID,下面:本文主要介绍使用雪花算法产生id导致前端精度缺失问题的解决方案,文中通过代... 目录一、问题根源二、解决方案1. 全局配置Jackson序列化规则2. 实体类必须使用Long封装类3.

Nginx部署React项目时重定向循环问题的解决方案

《Nginx部署React项目时重定向循环问题的解决方案》Nginx在处理React项目请求时出现重定向循环,通常是由于`try_files`配置错误或`root`路径配置不当导致的,本文给大家详细介... 目录问题原因1. try_files 配置错误2. root 路径错误解决方法1. 检查 try_f

在React聊天应用中实现图片上传功能

《在React聊天应用中实现图片上传功能》在现代聊天应用中,除了文字和表情,图片分享也是一个重要的功能,本文将详细介绍如何在基于React的聊天应用中实现图片上传和预览功能,感兴趣的小伙伴跟着小编一起... 目录技术栈实现步骤1. 消息组件改造2. 图片预览组件3. 聊天输入组件改造功能特点使用说明注意事项

一文详解如何在Vue3中封装API请求

《一文详解如何在Vue3中封装API请求》在现代前端开发中,API请求是不可避免的一部分,尤其是与后端交互时,下面我们来看看如何在Vue3项目中封装API请求,让你在实现功能时更加高效吧... 目录为什么要封装API请求1. vue 3项目结构2. 安装axIOS3. 创建API封装模块4. 封装API请求

全解析CSS Grid 的 auto-fill 和 auto-fit 内容自适应

《全解析CSSGrid的auto-fill和auto-fit内容自适应》:本文主要介绍了全解析CSSGrid的auto-fill和auto-fit内容自适应的相关资料,详细内容请阅读本文,希望能对你有所帮助... css  Grid 的 auto-fill 和 auto-fit/* 父元素 */.gri

Springboot实现推荐系统的协同过滤算法

《Springboot实现推荐系统的协同过滤算法》协同过滤算法是一种在推荐系统中广泛使用的算法,用于预测用户对物品(如商品、电影、音乐等)的偏好,从而实现个性化推荐,下面给大家介绍Springboot... 目录前言基本原理 算法分类 计算方法应用场景 代码实现 前言协同过滤算法(Collaborativ

HTML5中的Microdata与历史记录管理详解

《HTML5中的Microdata与历史记录管理详解》Microdata作为HTML5新增的一个特性,它允许开发者在HTML文档中添加更多的语义信息,以便于搜索引擎和浏览器更好地理解页面内容,本文将探... 目录html5中的Mijscrodata与历史记录管理背景简介html5中的Microdata使用M