首页 > 编程语言 >Vue中的虚拟DOM和Diff算法

Vue中的虚拟DOM和Diff算法

时间:2023-07-26 09:36:05浏览次数:55  
标签:Vue undefined DOM text vnode 虚拟 Diff 节点

一、 虚拟DOM

1. 什么是虚拟DOM?

一个用来表示真实 DOM 节点 的 JS 对象,主要包含标签名 tag、属性 attrs 和子元素对象 children 属性等。

代码示例如下:

<div class="contain" id="baseNo">
  <h4 class="item">标题</h4>
  <p class="item">段落内容</p>
</div>

{
  tag:'div',
  attrs:{
    id:'baseNo',
    class:'contain'
  },
  children:[
     {
      tag:'h4',
      attrs:{
        class:'item'
      },
      text:'标题'
    }
    {
      tag:'p',
      attrs:{
        class:'item'
      },
      text:'段落内容'
    },
  ]

}

2. 虚拟 DOM 存在的意义

大家都知道真实 DOM 是一棵节点树,在渲染时需要很大的性能开销,如果每次数据发生变化时,都直接渲染到真实 DOM 上会引起整个 DOM 树的重绘和重排,而通过虚拟 DOM-Diff 算法只需要在数据发生变化时,对新旧虚拟 DOM 进行差异化的比较,然后对真实 DOM 进行局部的更新操作,大大较少了渲染性能的开销,从而提高页面的渲染性能和用户体验。

3. 虚拟 DOM 具体实现

前面我们介绍了虚拟 DOM 的概念以及虚拟 DOM 存在的意义,那么在 Vue 中虚拟 DOM 是怎么实现的呢?接下来,我们从源码出发,深入学习一下

3.1 VNode 类

虚拟 DOM 是用 JS 对象来描述一个真实的 DOM 节点。而在 Vue 中就存在了一个 VNode 类,VNode 类中包含了描述一个真实 DOM 节点所需要的一系列属性,如 tag 表示节点的标签名,text 表示节点中包含的文本,children 表示该节点包含的子节点等。源码如下:

export default class VNode {
  tag?: string/*当前节点的标签名*/
  data: VNodeData | undefined/*当前节点对应的数据对象,包含了具体的一些数据信息,
  是一个VNodeData类型,可以参考VNodeData类型中的数据信息*/
  children?: Array<VNode> | null/*当前节点的子节点集合,是一个数组*/
  text?: string /*当前节点的文本*/
  elm: Node | undefined/*当前虚拟节点对应的真实dom节点*/
  ns?: string/*当前节点的名字空间*/
  context?: Component /*当前组件节点对应的Vue实例*/
  key: string | number | undefined/*节点的key属性,被当作节点的标志,用以优化*/
  componentOptions?: VNodeComponentOptions/*组件的option选项*/
  componentInstance?: Component /*当前节点对应的组件的实例*/
  parent: VNode | undefined | null /*当前节点的父节点*/

  // strictly internal
  raw: boolean /*简而言之就是是否为原生HTML或只是普通文本,
  innerHTML的时候为true,textContent的时候为false*/
  isStatic: boolean /*静态节点标志*/
  isRootInsert: boolean /*是否作为根节点插入*/
  isComment: boolean /*是否为注释节点*/
  isCloned: boolean  /*是否为克隆节点*/
  isOnce: boolean /*是否有v-once指令*/
  asyncFactory?: Function // async component factory function
  asyncMeta: Object | void
  isAsyncPlaceholder: boolean
  ssrContext?: Object | void
  fnContext: Component | void // /*函数式组件对应的Vue实例*/
  fnOptions?: ComponentOptions | null // for SSR caching
  devtoolsMeta?: Object | null // used to store functional render context for devtools
  fnScopeId?: string | null // functional scope id support
  isComponentRootElement?: boolean | null // for SSR directives

  constructor(
    tag?: string,
    data?: VNodeData,
    children?: Array<VNode> | null,
    text?: string,
    elm?: Node,
    context?: Component,
    componentOptions?: VNodeComponentOptions,
    asyncFactory?: Function
  ) {
    this.tag = tag
    this.data = data
    this.children = children
    this.text = text
    this.elm = elm
    this.ns = undefined
    this.context = context
    this.fnContext = undefined
    this.fnOptions = undefined
    this.fnScopeId = undefined
    this.key = data && data.key
    this.componentOptions = componentOptions
    this.componentInstance = undefined
    this.parent = undefined
    this.raw = false
    this.isStatic = false
    this.isRootInsert = true
    this.isComment = false
    this.isCloned = false
    this.isOnce = false
    this.asyncFactory = asyncFactory
    this.asyncMeta = undefined
    this.isAsyncPlaceholder = false
  }

  get child(): Component | void {
    return this.componentInstance
  }
}

通过属性之间不同的搭配,可以实例化出不同类型的真实 DOM 节点。

3.2 VNode 的类型

接下来我们通过阅读源码,对不同属性搭配出的以下几种类型节点进行分析

  1)注释节点

  2)文本节点

  3)克隆节点

3.2.1 注释节点

注释节点主要包含 text 和 isComment 俩个属性,其中 text 属性表示具体的注释信息, isComment 是一个标志,用来表示一个节点是否是注释节点。源码如下:

// 创建注释节点
export const createEmptyVNode = (text: string = '') => {
  const node = new VNode()
  node.text = text
  node.isComment = true
  return node
}
3.2.2 文本节点

文本节点只需要一个属性,那就是 text 属性,主要用来表示具体的文本内容。源码如下:

// 创建文本节点
export function createTextVNode (val: string | number) {
  return new VNode(undefined, undefined, undefined, String(val))
}
3.2.3 克隆节点

克隆节点就是把现有节点的属性全部复制到新节点中,而现有节点和新克隆得到的节点之间的不同在于克隆得到的节点中的 isCloned 值为 true,源码如下:

// 创建克隆节点
export function cloneVNode(vnode: VNode): VNode {
  const cloned = new VNode(
    vnode.tag,
    vnode.data,
    vnode.children,
    vnode.text,
    vnode.elm,
    vnode.context,
    vnode.componentOptions,
    vnode.asyncFactory
  )
  cloned.ns = vnode.ns
  cloned.isStatic = vnode.isStatic
  cloned.key = vnode.key
  cloned.isComment = vnode.isComment
  cloned.fnContext = vnode.fnContext
  cloned.fnOptions = vnode.fnOptions
  cloned.fnScopeId = vnode.fnScopeId
  cloned.asyncMeta = vnode.asyncMeta
  cloned.isCloned = true
  return cloned
}

以上几种节点类型它们本质上都是 VNode 类的实例,只是在实例化的时候传入的属性参数不同而已。

二、Diff 算法

1. 什么是 Diff 算法?

Diff 算法是一种高效的算法,可以在不重建所有节点的情况下,将新旧虚拟 DOM 比较,找出需要更新的节点,从而实现局部更新,提高应用程序的性能。Diff算法还约定只做同层级节点比对,而不是跨层级节点比对,即深度优先遍历算法,在同级之间则运用首尾指针法进行比较。

Diff 算法的作用:本质就是比较俩个 JS 对象的差异。

2. Diff 算法的实现原理

当数据修改后会触发 setter 劫持操作,我们在 setter 中执行 dep.notify(),通知所有的订阅者 watcher 重新渲染。 订阅者 watcher 这时会在回调内部,通过 vm._render()获取最新的虚拟 DOM;然后通过 patch 方法比对新旧虚拟 DOM,给真实 DOM 元素进行打补丁,更新视图的操作。 下面通过视图进行解析:

2.1 数据发生变化时,节点更新视图解析

image

2.2 节点进行比较时,同层级节点进行比对

image

2.3 updateChildren时, 首尾指针法视图解析

规则:

  1)依次比较,当比较成功后退出当前比较

  2)渲染结果以 newVnode 为准

  3)每次比较成功后 start 点和 end 点向中间靠拢

  4)当新旧节点中有一个 start 点跑到 end 点右侧时终止比较

  5)如果都匹配不到,则旧虚拟 DOM kev 值去比对 新虚拟 DOM 的 key 值,如果 key 相同则复用,并移动到新虚拟 DOM 的位置。

初始化:

  旧虚拟节点(oldVnode):首指针(oldSart),尾指针(oldEnd)

  新虚拟节点(newVnode):首指针(newSart),尾指针(newEnd)
image
第一步:首先,oldSart和newSart做比对,如果没有比对成功,就用oldSart和newEnd进行比对,当比对成功,就退出比对,将真实DOM中当前元素移至与新虚拟节点相同位置,并将oldS和newE指针向中间靠拢
imageimageimage

第二步:重复第一步操作,当oldVnode和newVnode其中有一个首指针跑到尾指针的右侧就终止比较,如oldVnode中有未匹配的元素,将其卸载;newVnode中有未匹配的,在真实DOM中相同位置新增即可。
imageimageimage
imageimageimage

三、总结

使用虚拟 DOM+Diff算法,主要是通过 JS 计算去替代对真实 DOM 的部分操作,最小化的更新视图,从而提高页面的渲染性能和用户体验。

标签:Vue,undefined,DOM,text,vnode,虚拟,Diff,节点
From: https://www.cnblogs.com/leayun/p/17581557.html

相关文章

  • 前端Vue仿支付宝自定义可滑动轮播分页宫格菜单组件,可支持九宫格 十二宫格 十五宫格
    背景介绍随着技术的不断发展,传统的开发方式使得系统的复杂度越来越高。在传统开发过程中,一个小小的改动或者一个小功能的增加可能会导致整体逻辑的修改,造成牵一发而动全身的情况。为了解决这个问题,我们采用了组件化的开发模式。通过组件化开发,可以有效地实现单独开发,单独维护,而且......
  • vue--day44-todolist的localStorage本地存储
    添加修改删除数据发生变化,可以用watch监测来实现监测数据的变化1.App.vue  <template><divid="root"><divclass="todo-container"><divclass="todo-wrap"><!--传递函数儿子给父亲传东西,父亲偷偷传递一个函数,儿子调用这个函数--><MyHeader:addTodo=&q......
  • Vue详解----一篇带你从头领悟到尾,享受飞升的感觉
    ......
  • 初学vue
    做为一个在圈子内混饭吃的我来说,对于vue是久有耳闻,但是混饭吃嘛,也不要太新的技术,一般实现客户需求就好,毕竟不是一线城市最近有一个项目需要用到Vue,而且计划给公司用uniapp开发一个手机端应用,里面也需要用到vue的相关技术,所以今天开始了学习之路初学嘛,当然跳过Vue2直接学vue3首......
  • vue-cats
    Vue-cats模板语法插值语法用于解析标签体内容{{xxx}}xxx是js表达式可以直接读取data的所有属性指令语法用于解析标签(包括:标签属性,标签体内容,绑定事件....)举例:v-bind:href="xxx"xxx也是js表达式数据动态绑定v-bind:简写为:数据绑定单向数......
  • 使用脚手架环境开发vue2项目
    一、检查node版本(1)有node.js(2)没有node.js安装地址:Node.js 二、使用npm,包管理工具,是node包管理和分发工具【需要配置仓库镜像地址,改为中国仓库镜像地址】1、查看当前镜像:npmgetregistry2、设置镜像地址的命令: 设置最新的国内淘宝镜像地址......
  • Vue 常用属性汇总
    1、Vue实例常用属性(1)数据data:Vue实例的数据对象components:Vue实例配置局部注册组件(2)类方法computed:计算属性watch:侦听属性filters:过滤器methods:Vue实例方法render:渲染函数,创建虚拟DOM(3)生命周期created:在实例创建完成后被立即调用,完成初始化操作mounted:el挂载到Vue实......
  • vue 项目打包iOS 配置
    Vue项目打包iOS配置Vue.js是一款用于构建用户界面的渐进式JavaScript框架。在开发过程中,我们经常需要将我们的Vue项目打包成iOS应用程序。本文将向您介绍如何配置一个Vue项目以打包成iOS应用程序,并提供代码示例供参考。安装依赖在开始之前,我们需要安装一些必要的......
  • vue redis
    实现VueRedis简介在开始讲解实现"VueRedis"之前,我们先来了解一下Redis是什么。Redis(RemoteDictionaryServer)是一个开源的内存数据结构存储系统,它常用于高性能的存储和数据缓存。Vue.js是一个用于构建用户界面的渐进式JavaScript框架。而"VueRedis"的目标是将Redis与Vue.js......
  • vue ios白屏
    如何解决VueiOS白屏问题简介在Vue开发过程中,偶尔会遇到iOS设备上页面白屏的问题。这个问题通常是由于iOS的Webview在加载Vue应用时出现的一些兼容性问题导致的。本文将介绍如何解决VueiOS白屏问题,并提供一些代码示例来帮助你完成解决方案。解决流程下面是解决VueiOS白屏问题......