广告
返回顶部
首页 > 资讯 > 精选 >Vue中Virtual DOM和Diff原理及实现方法是什么
  • 678
分享到

Vue中Virtual DOM和Diff原理及实现方法是什么

2023-07-05 14:07:17 678人浏览 泡泡鱼
摘要

本篇内容介绍了“Vue中Virtual DOM和Diff原理及实现方法是什么”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!1. v

本篇内容介绍了“Vue中Virtual DOM和Diff原理及实现方法是什么”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!

1. vdom

vdom即虚拟DOM,将DOM映射为js对象,结合diff算法更新DOM

以下为DOM

<div id="app">  <div class="home">home</div></div>

映射成VDOM

{  tag: 'div',  attrs: {    id: 'app'  },  children: [    {      tag: 'div',      attrs: {        class: 'home'      },      children: [        {          tag: undefined,          attrs: undefined,          text: 'home',          children: undefined        }      ]    }  ]}

通过这个vdom实现简单的render函数,可以通过js操作修改dom

<template>  <div id="app">    <div v-for="item in arr">{{ item.name }} : {{ item.id }}</div>  </div>  <button id="btn">reRender</button></template>
let app = document.getElementById('app')let data = {  arr: [       { name: 'a', id: 1 },    { name: 'b', id: 2 },    { name: 'c', id: 3 },  ]}function render(data) {  app.innerhtml = ''  let children = []  data.forEach(item => {    let el = document.createElement("div")    el.innerHtml = `${ item.name } : ${item.id}`    app.appendChild(el)  })  }// testrender(data.arr) // 首次渲染let btn = document.getElementById('btn')btn.onClick = () => {  data.arr[2].id++ // 修改关联数据  render(data.arr) // 重新渲染:暴力刷新DOM,没有diff,实际上只用更新最后一个div就行}

使用snabbdom实现VDOM

snabbldom是简易实现vdom功能的库,有两个核心api:h函数和patch函数

h(tag, attrs, children) // 创建vnodepatch(vnode, newVnode) // 对vnode进行diff后挂载到真实dom上

结合hpatch实现render渲染函数

let app = document.getElementById('app')let vnode;function render(data) {  let newVnode = h('div', { class: 'wrap' }, data.forEach(item => {      return h('div', {}, `${item.name} : ${item.id}`)    })  )  patch(vnode, newVnode)  vnode = newVnode}render(data.arr) // 首次渲染let btn = document.getElementById('btn')btn.onClick = () => {  data.arr[2].id++ // 修改关联数据  render(data.arr) // 重新渲染:在patch函数里经过vdom的diff后再挂载到真实dom,这里只更新最后一个div}

2. Diff

为了尽量减少DOM操作,需要通过diff对比新旧vnode,针对更改的地方进行更新DOM,而非替换整个DOM

大体思路为:

  • 对新旧两个节点调用patch函数

  • 进来先判断两个节点是否为同一类型,具体是对比keytagdata等属性

  • 若不为同一类型,那么基于新节点创建dom之后作替换

  • 若为同一类型,那么调用patchVnode函数

  • 进来先判断两个节点是文本节点的话,那么就作文本内容替换

  • 否则判断是否都有子节点,都有的话调用updateChildren函数,通过首尾四个指针对子节点数组进行diff更新;若旧节点有子节点,新节点没有,这时就删除子节点;若旧节点无子节点,新节点有,这时基于新节点创建dom作替换即可

通过createElment函数,将VDOM转为真实DOM

function createElement(vnode) {  if(vnode.text) return document.createTextNode(vnode) // 文本节点      let { tag, attrs, children } = vnode    let el = document.createElement(tag) // tag    for(let key of attrs){ // attrs    el.setAttribute(key, attrs[key])  }    children.forEach(childVnode => { // children    el.appendChild(createElement(childVnode))   })  vnode.el = el  return el}

通过patch函数,执行diff更新操作

判断vnodenewVnode是否为同一类型节点,是则继续递归对比子节点,否则直接替换

function patch(vnode, newVnode) {  if (isSameNode(vnode, newVnode)) patchVnode(vnode, newVnode)  else replaceVnode(vnode, newVnode)}function replaceVnode(vnode, newVnode) {  let el = vnode.el // 旧节点  let parentEl = api.getParentNode(el) // 获取父节点  api.insertBefore(parentEl, createElement(newVnode), api.getNextSibling(el)) // 插入新节点  api.removeChild(parentEl, el) // 删除旧节点}function isSameNode(vnode, newVnode) {  return (    vnode.key == newVnode.key && // key是否相同    vnode.tag == newVnode.tag && // tag是否相同    isDef(vnode.data) == isDef(newVnode.data) // 是否都定义了data    // &&... 其他条件  )}function patchVnode(vnode, newVnode) {  let el = newVnode.el = vnode.el // 获取当前旧节点对应的dom,并赋值给新节点的el  // 1.都为文本节点,且文本不一样  if (vnode.text && newVnode.text && vnode.text != newVnode.text)    return api.setElText(el, newVnode.text) // 替换文本    let ch = vnode.children  let newCh = newVnode.children  if (ch && newCh) return updateChildren(el, ch, newCh) // 2.都有子节点,递归对比  if (ch) return api.removeChild(el) // 3.vnode有子节点,newVnode无,删除子节点  return replaceVnode(vnode, newVnode) // 4. newNode有子节点,vnode无,替换即可}

updateChildren实现比较复杂,使用首尾四指针进行vnodenewVnode的对比

function updateChildren(el, ch, newCh) {  // 子节点下标  let l = 0  let r = ch.length - 1  let newL = 0  let newR = newCh.length - 1  // 子节点  let lNode = ch[l]  let rNode = ch[r]  let newLNode = newCh[newL]  let newRNode = newCh[newR]  while (l <= r && newL <= newR) {    if (!lNode || !rNode || !newLNode || !newRNode) { // 边界处理      if (!lNode) lNode = ch[++l]      if (!rNode) rNode = ch[--r]      if (!newLNode) newLNode = newCh[++newL]      if (!newRNode) newRNode = newCh[--newR]      continue    }        // 新旧子节点首尾指针对比 l*newL、r*newR、l*newR、r*newL    if (isSameNode(lNode, newLNode)) {      patchVnode(lNode, newLNode)      lNode = ch[++l]      newLNode = newCh[++newL]      continue    }    if (isSameNode(rNode, newRNode)) {      patchVnode(rNode, newRNode)      rNode = ch[--r]      newRNode = newCh[--newR]      continue    }    if (isSameNode(lNode, newRNode)) {      patchVnode(lNode, newRNode)      api.insertBefore(el, lNode.el, api.nextSibling(rNode.el))      lNode = ch[++l]      newRNode = newCh[--newR]      continue    }    if (isSameNode(rNode, newLNode)) {      patchVnode(rNode, newLNode)      api.insertBefore(el, rNode.el, lNode.el)      rNode = ch[--r]      newLNode = newCh[++newL]      continue    }    // 在vnode未知序列区间[l,r]生成key-idx的map表,用newLNode的key在未知序列中找到可复用的位置    if (!keyIdxMap) keyIdxMap = geTKEyIdxMap(ch, l, r) // map    keyIdx = keyIdxMap.get(newLNode.key)    if (!keyIdx) {      api.insertBefore(el, createElement(newLNode), lNode.el)    }    else {      let nodeToMove = ch[keyIdx]      patchVnode(nodeToMove, newLNode)      api.insertBefore(el, nodeToMove.el, lNode.el)    }    newLNode = newCh[++newL]  }}function getKeyIdxMap(ch, l, r) {  let map = new Map()  while (l <= r) map.set(ch[l].key, l++)  return map}

“Vue中Virtual DOM和Diff原理及实现方法是什么”的内容就介绍到这里了,感谢大家的阅读。如果想了解更多行业相关的知识可以关注编程网网站,小编将为大家输出更多高质量的实用文章!

--结束END--

本文标题: Vue中Virtual DOM和Diff原理及实现方法是什么

本文链接: https://www.lsjlt.com/news/352424.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • Vue中Virtual DOM和Diff原理及实现方法是什么
    本篇内容介绍了“Vue中Virtual DOM和Diff原理及实现方法是什么”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!1. v...
    99+
    2023-07-05
  • 浅析Vue中Virtual DOM和Diff原理及实现
    目录0. 写在开头1. vdom2. Diff0. 写在开头 本文将秉承Talk is cheap, show me the code原则,做到文字最精简,一切交由代码说明! 1. ...
    99+
    2023-03-21
    Vue Virtual DOM Diff原理 Vue Virtual DOM Vue Diff
  • MD5算法原理及C#和JS实现的方法是什么
    本篇内容主要讲解“MD5算法原理及C#和JS实现的方法是什么”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“MD5算法原理及C#和JS实现的方法是什么”吧!一、简介MD5 是哈希算法(散列算法)的...
    99+
    2023-07-05
  • cdn加速原理及实现方法是什么
    CDN加速原理是通过在全球分布的节点服务器上缓存静态资源,将用户请求转发至最近的缓存节点,从而提高用户访问速度和稳定性。CDN实现方...
    99+
    2023-05-30
    cdn加速原理 cdn
  • SHA-256算法原理及C#和JS实现的方法是什么
    本篇内容主要讲解“SHA-256算法原理及C#和JS实现的方法是什么”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“SHA-256算法原理及C#和JS实现的方法是什么”吧!一、简介SHA-256 ...
    99+
    2023-07-05
  • C++中内存池的原理及实现方法是什么
    这篇文章主要讲解了“C++中内存池的原理及实现方法是什么”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“C++中内存池的原理及实现方法是什么”吧!为什么要用内存池C++程序默认的内存管理(ne...
    99+
    2023-07-05
  • Java中​HashMap的工作原理及实现方法是什么
    今天小编给大家分享一下Java中HashMap的工作原理及实现方法是什么的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。Has...
    99+
    2023-06-03
  • DES&3DES算法原理及C#和JS实现的方法是什么
    这篇文章主要介绍“DES&3DES算法原理及C#和JS实现的方法是什么”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“DES&3DES算法原理及C#和JS实现的方法是什么”文章能帮助大...
    99+
    2023-07-05
  • Java动态代理的原理及实现方法是什么
    本篇内容主要讲解“Java动态代理的原理及实现方法是什么”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Java动态代理的原理及实现方法是什么”吧!代理是指:某些场景下对象会找一个代理对象,来辅助...
    99+
    2023-07-02
  • Vue中Watcher和Scheduler的实现原理是什么
    这篇文章主要介绍“Vue中Watcher和Scheduler的实现原理是什么”,在日常操作中,相信很多人在Vue中Watcher和Scheduler的实现原理是什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答...
    99+
    2023-06-21
  • 在vue中nextTick用法及nextTick的原理是什么
    目录什么是 nextTicknextTick 的用法nextTick 的实现原理总结Vue.js 是一个流行的前端框架,它提供了一种响应式的数据绑定机制,使得页面的数据与页面的 UI...
    99+
    2023-05-16
    vue nextTick使用 vue nextTick原理
  • PHP单例模式的原理及实现方法是什么
    本篇内容介绍了“PHP单例模式的原理及实现方法是什么”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!单例模式Singleton Pattern...
    99+
    2023-07-05
  • Spring 中 load 方法的实现原理是什么?
    Spring 是一个流行的 Java 开发框架,它提供了很多方便的功能,其中之一是 load 方法。load 方法可以根据指定的类型和 ID 加载一个对象。在本文中,我们将深入探讨 Spring 中 load 方法的实现原理。 load ...
    99+
    2023-10-15
    load spring 编程算法
  • CRC校验原理及C语言实现的方法是什么
    这篇“CRC校验原理及C语言实现的方法是什么”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“CRC校验原理及C语言实现的方法是...
    99+
    2023-07-05
  • go原子操作的方式及实现原理是什么
    今天小编给大家分享一下go原子操作的方式及实现原理是什么的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。什么是原子操作?原子操...
    99+
    2023-07-06
  • vue中v-model的实现原理是什么
    这篇文章主要介绍vue中v-model的实现原理是什么,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!v-model的实现原理是什么?为什么要使用VueVue是一款友好的、多用途且高性...
    99+
    2022-10-19
  • 微信域名检测API的实现方法及原理是什么
    微信域名检测API的实现方法及原理是什么,针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。微信域名检测API接口是腾讯对外公布的微信域名状态查询接口,可实时查询域名在微信中的状...
    99+
    2023-06-04
  • C++中线程的原理与实现方法是什么
    这篇文章主要介绍“C++中线程的原理与实现方法是什么”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“C++中线程的原理与实现方法是什么”文章能帮助大家解决问题。在C++中有多种实现线程的方式C++11...
    99+
    2023-07-05
  • JUC中wait与notify方法的实现原理是什么
    今天小编给大家分享一下JUC中wait与notify方法的实现原理是什么的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。1.O...
    99+
    2023-07-05
  • python中k-means和k-means++原理是什么及怎么实现
    这篇文章主要介绍“python中k-means和k-means++原理是什么及怎么实现”,在日常操作中,相信很多人在python中k-means和k-means++原理是什么及怎么实现问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作...
    99+
    2023-06-30
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作