为何出现:html
浏览器解析一个html大体分为五步:建立DOM tree –> 建立Style Rules -> 构建Render tree -> 布局Layout –> 绘制Painting。每次对真实dom进行操做的时候,浏览器都会从构建dom树开始从头至尾执行一遍流程。真实的dom操做代价昂贵,操做频繁还会引发页面卡顿影响用户体验,虚拟dom就是为了解决这个浏览器性能问题才被创造出来vue
虚拟dom在执行dom的更新操做后,虚拟dom不会直接操做真实dom,而是将更新的diff内容保存到本地js对象中,而后一次性attach到dom树上,通知浏览器进行dom绘制避免大量无谓的计算。node
如何实现:git
js对象表示dom结构,对象记录了dom节点的标签、属性和子节点github
js对象的render函数经过对虚拟dom的属性和子节点的递归构建出真实dom树算法
虚拟DOM是一个纯粹的JS对象,能够经过
document.createDocumentFragment
建立,Vue中一个虚拟DOM包含如下属性:api
- tag: 当前节点的标签名
- data: 当前节点的数据对象
- children: 数组类型,包含了当前节点的子节点
- text: 当前节点的文本,通常文本节点或注释节点会有该属性
- elm: 当前虚拟节点对应的真实的dom节点
- context: 编译做用域
- functionalContext: 函数化组件的做用域
- key: 节点的key属性,用于做为节点的标识,有利于patch的优化
- sel: 节点的选择器
- componentOptions: 建立组件实例时会用到的选项信息
- child: 当前节点对应的组件实例
- parent: 组件的占位节点
- raw: raw html
- isStatic: 静态节点的标识
- isRootInsert: 是否做为根节点插入,被包裹的节点,该属性的值为false
- isComment: 当前节点是不是注释节点
- isCloned: 当前节点是否为克隆节点
- isOnce: 当前节点是否有v-once指令
简单总结:虚拟DOM是将真实的DOM节点用JavaScript模拟出来,将DOM变化的对比,放到 Js 层来作。数组
diff算法是一种优化手段,将先后两个模块进行差别对比,修补(更新)差别的过程叫作patch浏览器
patch:bash
虚拟DOM最核心的部分,它能够将vnode渲染成真实的DOM,这个过程是对比新旧虚拟节点之间有哪些不一样,而后根据对比结果找出须要更新的的节点进行更新。
patch自己就有补丁、修补的意思,其实际做用是在现有DOM上进行修改来实现更新视图的目的。Vue的Virtual DOM patching算法是基于Snabbdom的实现,并在些基础上做了不少的调整和改进。
当数据发生改变时,set方法会让调用Dep.notify
通知全部订阅者Watcher,订阅者就会调用patch
给真实的DOM打补丁,更新相应的视图。
Vue的diff算法是仅在同级的vnode间作diff,递归地进行同级vnode的diff,最终实现整个DOM树的更新。由于跨层级的操做是很是少的,忽略不计,这样时间复杂度就从O(n3)变成O(n)。
当新旧虚拟节点的key和sel都相同时,则进行节点的深度patch,若不相同则整个替换虚拟节点,同时建立真实DOM,实现视图更新。
如何断定新旧节点是否为同一节点:
当两个VNode的tag、key、isComment都相同,而且同时定义或未定义data的时候,且若是标签为input则type必须相同。这时候这两个VNode则算sameVnode,能够直接进行patchVnode操做。
function patch (oldVnode, vnode) {
if (sameVnode(oldVnode, vnode)) { // 有必要进行patch, key和sel都相同时才进行patch
patchVnode(oldVnode, vnode)
} else { // 没有必要进行patch, 整个替换
const oEl = oldVnode.el
let parentEle = api.parentNode(oEl)
createEle(vnode) // vnode建立它的真实dom,令vnode.el =真实dom
if (parentEle !== null) {
api.insertBefore(parentEle, vnode.el, api.nextSibling(oEl)) // 插入整个新节点树
api.removeChild(parentEle, oldVnode.el) // 移出整个旧的虚拟DOM
oldVnode = null
}
}
return vnode
}
复制代码
深度patch:
function patchVnode (oldVnode, vnode, insertedVnodeQueue, removeOnly) {
/*两个VNode节点相同则直接返回*/
if (oldVnode === vnode) {
return
}
// reuse element for static trees.
// note we only do this if the vnode is cloned -
// if the new node is not cloned it means the render functions have been
// reset by the hot-reload-api and we need to do a proper re-render.
/* 若是新旧VNode都是静态的,同时它们的key相同(表明同一节点), 而且新的VNode是clone或者是标记了once(标记v-once属性,只渲染一次), 那么只须要替换elm以及componentInstance便可。 */
if (isTrue(vnode.isStatic) &&
isTrue(oldVnode.isStatic) &&
vnode.key === oldVnode.key &&
(isTrue(vnode.isCloned) || isTrue(vnode.isOnce))) {
vnode.elm = oldVnode.elm
vnode.componentInstance = oldVnode.componentInstance
return
}
let i
const data = vnode.data
if (isDef(data) && isDef(i = data.hook) && isDef(i = i.prepatch)) {
/*i = data.hook.prepatch,若是存在的话,见"./create-component componentVNodeHooks"。*/
i(oldVnode, vnode)
}
const elm = vnode.elm = oldVnode.elm
const oldCh = oldVnode.children
const ch = vnode.children
if (isDef(data) && isPatchable(vnode)) {
/*调用update回调以及update钩子*/
for (i = 0; i < cbs.update.length; ++i) cbs.update[i](oldVnode, vnode)
if (isDef(i = data.hook) && isDef(i = i.update)) i(oldVnode, vnode)
}
/*若是这个VNode节点没有text文本时*/
if (isUndef(vnode.text)) {
if (isDef(oldCh) && isDef(ch)) {
/*新老节点均有children子节点,则对子节点进行diff操做,调用updateChildren*/
if (oldCh !== ch) updateChildren(elm, oldCh, ch, insertedVnodeQueue, removeOnly)
} else if (isDef(ch)) {
/*若是老节点没有子节点而新节点存在子节点,先清空elm的文本内容,而后为当前节点加入子节点*/
if (isDef(oldVnode.text)) nodeOps.setTextContent(elm, '')
addVnodes(elm, null, ch, 0, ch.length - 1, insertedVnodeQueue)
} else if (isDef(oldCh)) {
/*当新节点没有子节点而老节点有子节点的时候,则移除全部ele的子节点*/
removeVnodes(elm, oldCh, 0, oldCh.length - 1)
} else if (isDef(oldVnode.text)) {
/*当新老节点都无子节点的时候,只是文本的替换,由于这个逻辑中新节点text不存在,因此直接去除ele的文本*/
nodeOps.setTextContent(elm, '')
}
} else if (oldVnode.text !== vnode.text) {
/*当新老节点text不同时,直接替换这段文本*/
nodeOps.setTextContent(elm, vnode.text)
}
/*调用postpatch钩子*/
if (isDef(data)) {
if (isDef(i = data.hook) && isDef(i = i.postpatch)) i(oldVnode, vnode)
}
}
复制代码
patchVnode的规则
1.若是新旧VNode都是静态的,同时它们的key相同(表明同一节点),那么只须要替换elm以及componentInstance便可(原地复用)。
2.新老节点均有children子节点且不一样,则对子节点进行diff操做,调用updateChildren,这个updateChildren也是diff的核心。
3.若是只有新节点存在子节点,先清空老节点DOM的文本内容,而后为当前DOM节点加入子节点。
4.若是只有老节点有子节点,直接删除老节点的子节点。
5.当新老节点都无子节点的时候,只是文本的替换。
接下来就是最复杂的diff算法的理解
updateChildren (parentElm, oldCh, newCh) {
let oldStartIdx = 0, newStartIdx = 0
let oldEndIdx = oldCh.length - 1
let oldStartVnode = oldCh[0]
let oldEndVnode = oldCh[oldEndIdx]
let newEndIdx = newCh.length - 1
let newStartVnode = newCh[0]
let newEndVnode = newCh[newEndIdx]
let oldKeyToIdx
let idxInOld
let elmToMove
let before
while (oldStartIdx <= oldEndIdx && newStartIdx <= newEndIdx) {
if (oldStartVnode == null) { // 对于vnode.key的比较,会把oldVnode = null
oldStartVnode = oldCh[++oldStartIdx]
}else if (oldEndVnode == null) {
oldEndVnode = oldCh[--oldEndIdx]
}else if (newStartVnode == null) {
newStartVnode = newCh[++newStartIdx]
}else if (newEndVnode == null) {
newEndVnode = newCh[--newEndIdx]
}else if (sameVnode(oldStartVnode, newStartVnode)) {
patchVnode(oldStartVnode, newStartVnode)
oldStartVnode = oldCh[++oldStartIdx]
newStartVnode = newCh[++newStartIdx]
}else if (sameVnode(oldEndVnode, newEndVnode)) {
patchVnode(oldEndVnode, newEndVnode)
oldEndVnode = oldCh[--oldEndIdx]
newEndVnode = newCh[--newEndIdx]
}else if (sameVnode(oldStartVnode, newEndVnode)) {
patchVnode(oldStartVnode, newEndVnode)
api.insertBefore(parentElm, oldStartVnode.el, api.nextSibling(oldEndVnode.el))
oldStartVnode = oldCh[++oldStartIdx]
newEndVnode = newCh[--newEndIdx]
}else if (sameVnode(oldEndVnode, newStartVnode)) {
patchVnode(oldEndVnode, newStartVnode)
api.insertBefore(parentElm, oldEndVnode.el, oldStartVnode.el)
oldEndVnode = oldCh[--oldEndIdx]
newStartVnode = newCh[++newStartIdx]
}else {
// 使用key时的比较
if (oldKeyToIdx === undefined) {
oldKeyToIdx = createKeyToOldIdx(oldCh, oldStartIdx, oldEndIdx) // 有key生成index表
}
idxInOld = oldKeyToIdx[newStartVnode.key]
if (!idxInOld) {
api.insertBefore(parentElm, createEle(newStartVnode).el, oldStartVnode.el)
newStartVnode = newCh[++newStartIdx]
}
else {
elmToMove = oldCh[idxInOld]
if (elmToMove.sel !== newStartVnode.sel) {
api.insertBefore(parentElm, createEle(newStartVnode).el, oldStartVnode.el)
}else {
patchVnode(elmToMove, newStartVnode)
oldCh[idxInOld] = null
api.insertBefore(parentElm, elmToMove.el, oldStartVnode.el)
}
newStartVnode = newCh[++newStartIdx]
}
}
}
if (oldStartIdx > oldEndIdx) {
before = newCh[newEndIdx + 1] == null ? null : newCh[newEndIdx + 1].el
addVnodes(parentElm, before, newCh, newStartIdx, newEndIdx)
}else if (newStartIdx > newEndIdx) {
removeVnodes(parentElm, oldCh, oldStartIdx, oldEndIdx)
}
}
复制代码
Vnode
的子节点Vch
和oldVnode
的子节点oldCh
提取出来oldCh
和vCh
各有两个头尾的变量StartIdx
和EndIdx
,它们的2个变量相互比较,一共有4种比较方式,,当其中两个能匹配上那么真实dom中的相应节点会移到Vnode相应的位置。若是4种比较都没匹配,若是设置了key
,就会用key
进行比较,在比较的过程当中,变量会往中间靠,一旦StartIdx>EndIdx
代表oldCh
和vCh
至少有一个已经遍历完了,就会结束比较。在新老两个VNode节点的左右头尾两侧都有一个变量标记,在遍历过程当中这几个变量都会向中间靠拢。当oldStartIdx <= oldEndIdx或者newStartIdx <= newEndIdx时结束循环。
咱们经过一个例子来理解整个对比过程:
真实节点:a,b,d
旧节点:a,b,d
新节点:a,c,d,b
第一步:
oldS = a, oldE = d;
S = a, E = b;
复制代码
oldS和S,E比较;oldE和S,E比较,得出oldS
和S
匹配的结论,因而a节点应该按照新节点的顺序放置在第一个。此时旧节点的a节点也在第一个,故而位置不动;
第一轮对比结束oldS和S为同一节点,向后移动,oldE和E不动;
第二步:
旧节点:a,b,d
新节点:a,c,d,b
oldS = b, oldE = d;
S = c, E = b;
复制代码
四个变量两辆对比可得oldS
和E
匹配,将本来的b节点移动到最后,由于E
是最后一个节点,他们位置要一致,这就是上面说的:当其中两个能匹配上那么真实dom中的相应节点会移到Vnode相应的位置;
第二轮对比结束,oldE和E为同一节点,向前移动,oldS和S位置不动;
第三步:
旧节点:a,d,b
新节点:a,c,d,b
oldS = d, oldE = d;
S = c, E = d;
复制代码
oldE
和E
匹配,位置不变;
第四步:
旧节点:a,d,b
新节点:a,c,d,b
oldS++;
oldE--;
oldS > oldE;
复制代码
遍历结束,说明旧节点先遍历完。就将剩余的新节点c根据本身的的index插入到真实dom中去
旧节点:a,c,d,b
新节点:a,c,d,b
对比完成。
固然也会存在四个变量没法互相匹配,分为两种状况
S
的key与hash表作匹配,匹配成功就判断S
和匹配节点是否为sameNode
,若是是,就在真实dom中将成功的节点移到最前面,不然,将S
生成对应的节点插入到dom中对应的oldS
位置,oldS
和S
指针向中间移动。S
生成新的节点插入真实DOM
(这里能够解释为何设置key会让diff更高效结束时存在两种具体的状况:
oldS > oldE
,能够认为旧节点先遍历完。固然也有可能新节点此时也正好完成了遍历,统一都归为此类。此时S和E之间的vnode是新增的,调用addVnodes,把这些虚拟node.elm所有插进before的后边.
S> E
,能够认为新节点先遍历完。此时oldS和oldE之间的节点在新的子节点里已经不存在了,直接删除
在模拟两个例子体会一下
eg.1
O b,a,d,f,e
N a,b,e
1.
oldS = b, oldE = e;
S = a, E = e;
O b,a,d,f,e
N a,b,e
2.
oldS = b, oldE = f;
S = a, E = b;
O a,d,f,b,e
N a,b,e
3.
s>e d,f 删除
O a,b,e
N a,b,e
复制代码
eg.2
O b,d,c,a
N a,e,b,f
1.
oldS = b, oldE = a;
S = a, E = f;
O a,b,d,c
N a,e,b,f
2.
oldS = d, oldE = c;
S = e, E = f;
此时四个参数没法匹配,根据key来对比O中是否有S对应的节点,没有,则在O的S位置插入对应节点
O a,e,d,b,c
N a,e,b,f
3.
oldS = d, oldE = c;
S = b, E = f;
此时四个参数没法匹配,根据key查找是否有S对应的B节点,有,移动到S当前的位置
O a,e,b,d,c
N a,e,b,f
4.
oldS = d, oldE = c;
S = f, E = f;
此时四个参数没法匹配,根据key查找是否有S对应的f节点,没有,则在O的S位置插入对应节点
O a,e,b,d,c,f
N a,e,b,f
5.
oldS = d, oldE = c;
s>f
循环结束,oldS与oldE之间的节点删除
复制代码
总结:
参考: