虚拟DOM
操作DOM的代价
操作DOM的代价很高
,影响页面性能的主要问题有以下几点:
- 访问和修改DOM元素
- 修改DOM元素的样式,导致
重绘
或重排
- 通过对DOM元素的事件处理,完成与用户的交互功能
DOM的修改会导致重绘或重排
- 重绘:重绘是指一些样式的修改,元素的位置和大小都没有改变,浏览器会根据元素的新属性重新绘制,使元素呈现新的外观。
- 重排/回流:是指元素的位置或尺寸发生了变化,浏览器需要重新计算渲染树,而新的渲染树建立后,浏览器会重新绘制页面。
重绘相对于重排还好一些,重绘仅仅改变变化元素的样式即可,但重排(回流)则会重新计算所有元素之间的位置关系然后重新绘制元素
如果频繁操作DOM,其必然带来性能变低,浏览器卡慢
为何需要虚拟DOM?
接下来,我们看一下真实的DOM元素,我们打开某度的首页,在控制台输入以下代码:
1 | var dom1 = document.querySelectorAll('div')[0] |
可以看到一个div下其实是有很多属性的:
1 | align |
一个DOM拥有这么多属性,这也是带来性能问题的原因之一,撇开我们用不上的属性,我们其实可以使用js来模拟一个仅保留我们需要的属性的DOM
,这样的模拟DOM其实就是虚拟DOM
。
比如我们可以用以下代码模拟一个内容为’Hello Word’,id名与class名为test的div元素:
1 | { |
Vue、React都使用了虚拟DOM技术,让新、旧DOM的变化对比在Js层完成,最后仅修改变化了的DOM,直接避免了频繁操作DOM的情况,大大提升页面性能。
Vnode
Vnode就是虚拟DOM技术在Vue中的实现,它的源码在这里,它在模拟DOM的情况下又添加了很多框架本身需要的属性。
1 | export default class VNode { |
有了虚拟DOM那么就要有对比新、旧虚拟DOM的变化算法,这种算法就叫Diff算法:
Diff算法
Diff算法同级比较
求两个任意树之间的最小修改是一个时间复杂度为O(n^3)问题。这样的时间复杂度是我们无法接受的。
在Web应用中将组件移动到树中的不同级别是非常罕见的,通常只在孩子中间横向移动。
所以Diff算法采用的是同级比较
,将算法的时间复杂度降低到了O(N),这大大降低了复杂性,同时也不会造成很大损失,正如下图所示:
所以如果我们进行了跨级别的组件移动操作,实际上是会先删除DOM,再在对应的层级上新建一个DOM。
循环中为何需要key属性?
我们看这张图:
如果我们循环生成了5个组件,然后我们又插入了一个新的同类组件,对于我们来说很难知道如何在两个组件Lists中建立映射,所以就会变成上图左侧所示,按顺序一一建立关联。
如果有了key的存在情况则大不一样,它能很容易的帮助代码解决映射问题,让代码在正确的地方进行正确的操作,这对代码的性能提升也有很大的帮助。
简单分析Diff算法
我们以Vue(v2.6.8)代码为例,代码位置在src/core/vdom/patch.js中。
首先我们先明确几个方法:
工具方法isUndef、isDef等:
1
2
3
4
5
6
7
8
9// 判断v是否是undefined或null
export function isUndef (v: any): boolean %checks {
return v === undefined || v === null
}
// 判断v是否不是undefined或null
export function isDef (v: any): boolean %checks {
return v !== undefined && v !== null
}
// 其他工具方法可以自行查看sameVnode:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17// 判断是否是同一个Vnode
function sameVnode (a, b) {
return (
a.key === b.key && (
(
a.tag === b.tag &&
a.isComment === b.isComment &&
isDef(a.data) === isDef(b.data) &&
sameInputType(a, b)
) || (
isTrue(a.isAsyncPlaceholder) &&
a.asyncFactory === b.asyncFactory &&
isUndef(b.asyncFactory.error)
)
)
)
}nodeOps 封装了一些原生DOM操作方法,在platforms\web\runtime\node-ops.js中
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25// code...
export function createElementNS (namespace: string, tagName: string): Element {
return document.createElementNS(namespaceMap[namespace], tagName)
}
export function createTextNode (text: string): Text {
return document.createTextNode(text)
}
export function createComment (text: string): Comment {
return document.createComment(text)
}
export function insertBefore (parentNode: Node, newNode: Node, referenceNode: Node) {
parentNode.insertBefore(newNode, referenceNode)
}
export function removeChild (node: Node, child: Node) {
node.removeChild(child)
}
export function appendChild (node: Node, child: Node) {
node.appendChild(child)
}
// code...
patch
对比新、老vnode,进行最小程度的修改
- 如果是
初始化
会传以下几个参数(core\instance\lifecycle.js): vm.__patch__(vm.$el, vnode, hydrating, false)
- // vm.$el 是要挂载到的DOM,vnode就是vnode,hydrating用于服务端渲染不用管,最后一个参数是removeOnly
- 如果是
更新
会传两个参数 vm.__patch__(prevVnode, vnode)
- // prevVnode 是旧 vNode,vnode 是新 vNode
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120core\vdom\patch.js
return function patch (oldVnode, vnode, hydrating, removeOnly) {
// vnode不存在,oldVnode存在,说明节点被移除了,直接调用销毁钩子
if (isUndef(vnode)) {
if (isDef(oldVnode)) invokeDestroyHook(oldVnode)
return
}
let isInitialPatch = false
const insertedVnodeQueue = []
// 如果oldVnode不存在的话,就新建一个根节点
if (isUndef(oldVnode)) {
// empty mount (likely as component), create new root element
isInitialPatch = true
createElm(vnode, insertedVnodeQueue)
} else {
// 👇根据 oldVnode 是否存在 nodeType 属性 来判断是否是一个真实DOM节点
// 👇如果存在 nodeType 说明当前走的是 初始化 流程
const isRealElement = isDef(oldVnode.nodeType)
// 走update流程 且 是同一个节点,直接调用 patchVnode 方法
if (!isRealElement && sameVnode(oldVnode, vnode)) {
// 修补现有根节点
patchVnode(oldVnode, vnode, insertedVnodeQueue, null, null, removeOnly)
}
else {
// oldVnode 是 真实节点,走 init 流程
if (isRealElement) {
// mounting to a real element
// check if this is server-rendered content and if we can perform
// a successful hydration.
// Vnode在服务端渲染的一些处理,这里暂且不看
// 如果oldVnode的是一个Element节点 && 存在服务端渲染的属性
if (oldVnode.nodeType === 1 && oldVnode.hasAttribute(SSR_ATTR)) {
// 则移除其SSR属性,再将hydrating设置为true
oldVnode.removeAttribute(SSR_ATTR)
hydrating = true
}
if (isTrue(hydrating)) {
if (hydrate(oldVnode, vnode, insertedVnodeQueue)) {
invokeInsertHook(vnode, insertedVnodeQueue, true)
return oldVnode
} else if (process.env.NODE_ENV !== 'production') {
warn(
'The client-side rendered virtual DOM tree is not matching ' +
'server-rendered content. This is likely caused by incorrect ' +
'HTML markup, for example nesting block-level elements inside ' +
'<p>, or missing <tbody>. Bailing hydration and performing ' +
'full client-side render.'
)
}
}
// either not server-rendered, or hydration failed.
// create an empty node and replace it
// 不是服务端渲染的话,且是初始化流程,把oldVnode替换为一个空的vNode
oldVnode = emptyNodeAt(oldVnode)
}
// 当前节点与其父节点
const oldElm = oldVnode.elm
const parentElm = nodeOps.parentNode(oldElm)
// 创建一个新的 node
createElm(
vnode,
insertedVnodeQueue,
// extremely rare edge case: do not insert if old element is in a
// leaving transition. Only happens when combining transition +
// keep-alive + HOCs. (#4590)
oldElm._leaveCb ? null : parentElm,
nodeOps.nextSibling(oldElm)
)
// update parent placeholder node element, recursively
// 递归更新父节点占位节点元素
if (isDef(vnode.parent)) {
let ancestor = vnode.parent
const patchable = isPatchable(vnode)
while (ancestor) {
for (let i = 0; i < cbs.destroy.length; ++i) {
cbs.destroy[i](ancestor)
}
ancestor.elm = vnode.elm
if (patchable) {
for (let i = 0; i < cbs.create.length; ++i) {
cbs.create[i](emptyNode, ancestor)
}
// #6513
// invoke insert hooks that may have been merged by create hooks.
// e.g. for directives that uses the "inserted" hook.
const insert = ancestor.data.hook.insert
if (insert.merged) {
// start at index 1 to avoid re-invoking component mounted hook
for (let i = 1; i < insert.fns.length; i++) {
insert.fns[i]()
}
}
} else {
registerRef(ancestor)
}
ancestor = ancestor.parent
}
}
// 有父元素
if (isDef(parentElm)) {
removeVnodes(parentElm, [oldVnode], 0, 0)
}
// 没有父元素触发销毁
else if (isDef(oldVnode.tag)) {
invokeDestroyHook(oldVnode)
}
}
}
invokeInsertHook(vnode, insertedVnodeQueue, isInitialPatch)
return vnode.elm
}
patch 方法针对初始化
与更新
这两种情况做处理,
关于初始化
与更新
的判断:patch 函数的第一个参数传的如果是一个真实DOM,那么就会有nodeType属性,则是初始化。
如果是更新,且新旧两个vNode值得比较(即调用samevnode方法返回true,说明是同一个节点)则会调用 patchVnode 进一步比较。
patchVnode
修补vnode
1 | function patchVnode ( oldVnode, vnode, insertedVnodeQueue, ownerArray, index, removeOnly ) { |
这里主要看这段:
1 | if (isDef(oldCh) && isDef(ch)) { |
- 如果 oldVnode 与 vNode 都有 children 则调用
updateChildren
方法来对比他俩的 children,
在 updateChildren 方法中就会用到Diff算法
来对比、更新节点,同时再 updateChildren 中也会调用 patchVnode 继续对比下一级子节点。 - 如果oldVnode 没有 children,而 vNode 有,则调用 addVnode 方法,添加所有的 children。
- 如果 oldVnode 有 children,而 vNode 有,则调用 removeVnode 方法,移除原有的 children。
- 如果 oldVnode 与 vNode 都是文本节点,则会用 vNode 的文本替换 oldVnode 的文本。
updateChildren
这个方法是 diff 算法的核心:
1 | function updateChildren (parentElm, oldCh, newCh, insertedVnodeQueue, removeOnly) { |
updateChildren的代码中呢,主要是一个while循环,新旧Lists中无论哪一个先循环完都会退出循环
1 | while (oldStartIdx <= oldEndIdx && newStartIdx <= newEndIdx) { |
直接看源码感觉头都炸了,这里通过画图的方式会更合适一些。注意:以下每张图之间没有联系
首先会在新旧Lists的头尾定义各定义一个标记,分别为:oldStartIdx,oldEndIdx,newStartIdx,newEndIdx,用图表示是这个样子:
oldStartVnode, newStartVnode相同的情况:
执行patchVnode方法
oldStartVnode与newStartVnode都前进一格
完成这些操作就变成了下图这样:oldEndVnode, newEndVnode相同的情况:
执行patchVnode方法
oldEndVnode与newEndVnode都后退一格
完成这些操作就变成了下图这样:oldStartVnode, newEndVnode相同的情况:
这种情况下意味着当前旧Lists的StartIdx位置的元素
,在新Lists中
被挪到了EndIdx位置
(Vnode moved right)
在执行完patchVnode方法之后,在真实DOM中
我们还要将oldStart 插到 oldEnd之后
oldStartVnode前进一格
newEndVnode后退一格oldEndVnode, newStartVnode相同的情况:
这种情况下意味着当前旧Lists的EndIdx位置的元素
,在新Lists中
被挪到了StartIdx位置
(Vnode moved left)
在执行完patchVnode方法之后,在真实DOM中
我们还要将oldEnd 插到 oldStart之前
newStartVnode前进一格
oldEndVnode后退一格
ELSE!
如果上面四种情况都比对不中,也是就出现下图的情况:
则会执行 createKeyToOldIdx
方法,
1 | function createKeyToOldIdx (children, beginIdx, endIdx) { |
返回一个 哈希表(obj),各项 键为 vnode 的 key属性,值为 vnode 的下标
哈希表中的内容包含处于 oldStart 至 oldEnd 的 vnode,大概长这样:
1 | { |
接着从哈希表中寻找是否有与newStartVnode一致key
的oldVNode节点
接着看下面第5条:
- 我们在哈希表中找到了oldVnode节点:
1
vnodeToMove = oldCh[idxInOld] // 这个就是我们找到的`旧的vnode`
这里还分了两种情况
- 一、
光比较key,肯定不足以判断两个vnode相同
,我着这里再调用sameVnode(vnodeToMove, newStartVnode)方法来对比
如果相同:
执行patchVnode
oldCh[idxInOld]赋undefined // oldCh[idxInOld] = undefined ,我们已经用vnodeToMove保存了一份了
然后在真实DOM中
,把vnodeToMove插入到oldStart之前
newStartVnode都前进一格
放代码把:1
2
3
4
5
6vnodeToMove = oldCh[idxInOld]
if (sameVnode(vnodeToMove, newStartVnode)) {
patchVnode(vnodeToMove, newStartVnode, insertedVnodeQueue, newCh, newStartIdx)
oldCh[idxInOld] = undefined
canMove && nodeOps.insertBefore(parentElm, vnodeToMove.elm, oldStartVnode.elm)
} else {}
- 二、key相同,但sameVnode比较出来不相同
这种情况下则调用createElm创建一个新的元素插到oldStart前面
newStartVnode都前进一格
我们在哈希表中!!!没有找到oldVnode!!!节点:
这种情况下和 5 中的第二种情况一模一样
调用createElm创建一个新的元素插到oldStart前面
newStartVnode都前进一格到了这一步,while已经循环完毕了,接下来要处理新旧List长短不相同的情况
- 一、oldStartIdx > oldEndIdx,oldStart 超过了oldEnd,说明
新List比旧Lists长
我们需要把没遍历到的vnode选出来,用refElm存一下,然后啊,使用addVnodes批量调用创建(createElm)把这些vnode加到真实DOM中1
2
3
4if (oldStartIdx > oldEndIdx) {
refElm = isUndef(newCh[newEndIdx + 1]) ? null : newCh[newEndIdx + 1].elm
addVnodes(parentElm, refElm, newCh, newStartIdx, newEndIdx, insertedVnodeQueue)
}
- 二、newStartIdx > newEndIdx,说明
旧List比新Lists长
我们调用removeVnodes方法,参数包含oldStartIdx 与 oldEndIdx,把多余的删掉1
2
3
4
5
6if{
// code...
}
else if (newStartIdx > newEndIdx) {
removeVnodes(parentElm, oldCh, oldStartIdx, oldEndIdx)
}
参考
高频dom操作和页面性能优化探索
React’s diff algorithm
https://github.com/answershuto/learnVue