时间:2021-07-01 10:21:17 帮助过:9人阅读
本文介绍了vue 虚拟dom的patch源码分析,分享给大家,具体如下:
源码目录:src/core/vdom/patch.js
updateChildren
方法主要通过while
循环去对比2棵树的子节点来更新dom
,通过对比新的来改变旧的,以达到新旧统一的目的。
通过一个例子来模拟一下:
假设有新旧2棵树,树中的子节点分别为a,b,c,d
等表示,不同的代号代表不同的vnode
,如:
在设置好状态后,我们开始第一遍比较,此时oldStartVnode=a,newStartVnode=a;
命中了sameVnode(oldStartVnode,newStartVnode)
逻辑,则直接调用patchVnode(oldStartVnode,newStartVnode,insertedVnodeQueue)
方法更新节点a
,接着把oldStartIdx
和newStartIdx
索引分别+1,如图:
更新完节点a
后,我们开始第2遍比较,此时oldStartVnode=b,newEndVnode=b;
命中了sameVnode(oldStartVnode,newEndVnode)
逻辑,则调用patchVnode(oldStartVnode, newEndVnode, insertedVnodeQueue)
方法更新节点b
,接着调用canMove && nodeOps.insertBefore(parentElm, oldStartVnode.elm, nodeOps.nextSibling(oldEndVnode.elm))
,把节点b
移到树的最右边,最后把oldStartIdx
索引+1,newEndIdx
索引-1,如图:
更新完节点b
后,我们开始第三遍比较,此时oldEndVnode=d,newStartVnode=d;
命中了sameVnode(oldEndVnode, newStartVnode)
逻辑,则调用patchVnode(oldEndVnode, newStartVnode, insertedVnodeQueue)
方法更新节点d
,接着调用canMove && nodeOps.insertBefore(parentElm, oldEndVnode.elm, oldStartVnode.elm)
,把d
移到c
的左边。最后把oldEndIdx
索引-1,newStartIdx
索引+1,如图:
更新完d
后,我们开始第4遍比较,此时newStartVnode=e
,节点e
在旧树里是没有的,因此应该被作为一个新的元素插入,调用createElm(newStartVnode, insertedVnodeQueue, parentElm, oldStartVnode.elm)
,后面执行了nodeOps.insertBefore(parent, elm, ref)
方法把e
插入到c
之前,接着把newStartIdx
索引+1,如图:
插入节点e
后,我们可以看到newStartIdx
已经大于newEndIdx
了,while
循环已经完毕。接着调用removeVnodes(parentElm, oldCh, oldStartIdx, oldEndIdx)
删除旧的c
,最终如图:
updateChildren
通过以上几步操作完成了旧树子节点的更新,实际上只用了比较小的dom
操作,在性能上有所提升,并且当子节点越复杂,这种提升效果越明显。vnode
通过patch
方法生成dom
后,会调用mounted hook
,至此,整个vue
实例就创建完成了,当这个vue
实例的watcher
观察到数据变化时,会两次调用render
方法生成新的vnode
,接着调用patch
方法对比新旧vnode
来更新dom
.
上面是我整理给大家的,希望今后会对大家有帮助。
相关文章:
JQuery选中select组件被选中的值方法
vue.js中$set与数组更新方法_vue.js
vue与vue-i18n结合实现后台数据的多语言切换方法
以上就是在vue中实现虚拟dom的patch(详细教程)的详细内容,更多请关注Gxl网其它相关文章!