vue3从精通到入门2:虚拟DOM的生成与真实DOM的转化

虚拟 DOM 实现是 Vue 框架的核心部分之一,它负责在真实 DOM 之上抽象出一个轻量级的、可复用的 JavaScript 对象树,用于高效地更新视图。

什么是虚拟DOM?

虚拟 DOM 是一个编程概念,它将真实的 DOM 树抽象为一个轻量级的 JavaScript 对象树。当应用状态发生变化时,Vue 会先比较新的虚拟 DOM 树和旧的虚拟 DOM 树之间的差异,然后只更新这些差异部分对应的真实 DOM,而不是重新渲染整个页面。这种方式大大提高了渲染性能。

而我们如何将虚拟DOM转化成真实DOM呢?

App.vue:

html 复制代码
<template>
  <div>
    <a href="https://vitejs.dev" target="_blank">
      <img src="/vite.svg" class="logo" alt="Vite logo" />
    </a>
    <a href="https://vuejs.org/" target="_blank">
      <img src="./assets/vue.svg" class="logo vue" alt="Vue logo" />
    </a>
  </div>
  <HelloWorld msg="Vite + Vue" />
</template>

从我们编写的vue代码以及转换成真实DOM整个流程如下:

模板 > render函数 > 虚拟DOM > 真实DOM

上图中是我写的App.vue。我们将其打印出来

javascript 复制代码
import { createApp } from 'vue'
import './style.css'
import App from './App.vue'

console.log(App); // 打印出来的模板

createApp(App).mount('#app')

打印结果:

我们可以看到这个render函数,我们的虚拟DOM是由render函数创建的。

javascript 复制代码
export function render(_ctx, _cache, $props, $setup, $data, $options) {
  const _component_HelloWorld = _resolveComponent("HelloWorld")

  return (_openBlock(), _createElementBlock("template", null, [
    _createElementVNode("div", null, [
      _createElementVNode("a", {
        href: "https://vitejs.dev",
        target: "_blank"
      }, [
        _createElementVNode("img", {
          src: "/vite.svg",
          class: "logo",
          alt: "Vite logo"
        })
      ]),
      _createElementVNode("a", {
        href: "https://vuejs.org/",
        target: "_blank"
      }, [
        _createElementVNode("img", {
          src: "./assets/vue.svg",
          class: "logo vue",
          alt: "Vue logo"
        })
      ])
    ]),
    _createVNode(_component_HelloWorld, { msg: "Vite + Vue" })
  ]))
}

当我的虚拟DOM创建出来后,如何转成真实DOM呢?

这里用到了patch 函数:

patch是 Vue 中用于比较和更新虚拟 DOM 的核心函数。它接受两个参数:旧的 VNode 和新的 VNode。根据这两个节点的类型和属性等信息,patch 函数会决定是否需要更新真实 DOM,以及如何更新。

简单表示下patch相关作用!

javascript 复制代码
function patch(  
  n1: VNode | null,  // 旧虚拟DOM
  n2: VNode,  // 新的虚拟DOM
  container: HostNode,  
  anchor: ?HostNode = null,  
  parentComponent: ?Component = null,  
  parentSuspense: ?SuspenseBoundary = null,  
  isSVG: boolean = false,  
  optimized: boolean = false  
): VNode {  
  // ...  
  const { type, ref, shapeFlag } = n2;  
  
  switch (type) {  
    case Text:  
      // 处理文本节点  
      // ...  
      break;  
    case Comment:  
      // 处理注释节点  
      // ...  
      break;  
    case Static:  
      // 处理静态节点  
      // ...  
      break;  
    case Fragment:  
      // 处理 Fragment 节点  
      // ...  
      break;  
    default:  
      // 处理元素或组件节点  
      if (shapeFlag & ShapeFlags.ELEMENT) {  
        // ... 处理元素节点 ...  
      } else if (shapeFlag & ShapeFlags.COMPONENT) {  
        // ... 处理组件节点 ...  
      }  
      // ...  
  }  
  
  // ... 其他逻辑,如处理子节点、引用、挂载等 ...  
}

后面单开一章讲解patch!

相关推荐
意法半导体STM321 天前
【官方原创】FDCAN数据段波特率增加后发送失败的问题分析 LAT1617
javascript·网络·stm32·单片机·嵌入式硬件·安全
为什么不问问神奇的海螺呢丶1 天前
n9e categraf redis监控配置
前端·redis·bootstrap
云飞云共享云桌面1 天前
推荐一些适合10个SolidWorks设计共享算力的服务器硬件配置
运维·服务器·前端·数据库·人工智能
Liu.7741 天前
vue开发h5项目
vue.js
咔咔一顿操作1 天前
轻量无依赖!autoviwe 页面自适应组件实战:从安装到源码深度解析
javascript·arcgis·npm·css3·html5
刘联其1 天前
.net也可以用Electron开发跨平台的桌面程序了
前端·javascript·electron
韩曙亮1 天前
【jQuery】jQuery 选择器 ④ ( jQuery 筛选方法 | 方法分类场景 - 向下找后代、向上找祖先、同级找兄弟、范围限定查找 )
前端·javascript·jquery·jquery筛选方法
前端 贾公子1 天前
Node.js 如何处理 ES6 模块
前端·node.js·es6
pas1361 天前
42-mini-vue 实现 transform 功能
前端·javascript·vue.js
柒.梧.1 天前
从零搭建SpringBoot+Vue+Netty+WebSocket+WebRTC视频聊天系统
vue.js·spring boot·websocket