浅识vue的虚拟DOM和渲染器

news2025/3/1 4:06:12

虚拟DOM本质上是对DOM的抽象描述,就是一个普通的js对象。他身上的属性要比真实DOM的属性要少得多。

在一定情况下,使用虚拟DOM的性能要逊于直接使用真实DOM。

例如,在页面一开始的时候,Vue需要先通过生成虚拟DOM树,在根据虚拟DOM树创建真实DOM挂载到页面上,那么要比直接创建真实DOM进行挂载要多一步,理论上这里使用vue反倒会慢一些,但是几乎没有差别。

vue这个框架属于声明式代码,原生的属于命令式代码。

原生对比起框架,理论上性能一定比较好。

例如修改了一个DOM的文本,原生可以做到直接修改具体dom的文本,而框架需要通过diff找到最小更新量,然后进行修改。

但原生要做到极致的性能,往往花费的时间要更多,这无疑是性价比不高的做法。

并且原生写的代码难以维护,而框架写的代码不仅将很多重复的操作进行了结合,例如操作操作真实DOM,大大提高了我们的开发效率,也增强了代码的可维护性。

虚拟DOM的意义在于为Diff算法服务。

Vue在组件更新时,会生成新的虚拟DOM树,然后进行新旧两树的对比,完成更新。其中对比更新,不是进行真实DOM的一个更新,而是通过虚拟DOM进行比较更新。

我们简单的了解一下虚拟DOM的意义。

接下来我们用代码来简单说明一下虚拟DOM和渲染器。

这一部分不是“源码解读”,但可以让我们帮助我们增加对虚拟DOM渲染器的理解。

假设我们有这样一个模板

<div @click="()=>{console.log(123)}">
      123<img
        src="https://img0.baidu.com/it/u=1628473271,2485762845&fm=253&fmt=auto&app=138&f=JPEG?w=500&h=695"
        alt="这是一张图片"
      />
    </div>

那么用虚拟DOM怎么描述呢

const vnode = {
    tagName: "div",
    props: {
        onClick: () => {
            console.log(123);
        }
    },
    children : [
        "123",
        {
            tagName : "img",
            props : {
                src : "...",
                alt : "这是一张图片"
            }

        }
    ]
}

非常简单的一个js对象

为了简化这个对象的生成,我们创建一个类和一个函数来帮助我们。

首先创建一个虚拟DOM的一个类。

/** 简易化虚拟DOM */

export class Vnode  {
    constructor (tagName,props = {},children = []) {
        this.tagName = tagName;
        this.props = props;
        this.children = children;
    }
}

然后写一个虚拟DOM创建函数

// 这里写的其实就是vue render函数里面的h函数
// h函数就是一个辅助创建虚拟DOM的工具函数

export const createVnode = (tagName,props,children) => {
    return new Vnode(tagName,props,children);
}

这里的createVnode函数其实就是我们render函数的里的参数函数h。

那虚拟DOM创建好了,就需要通过渲染器生成真实DOM挂载到对应节点上。

在这里插入图片描述

mountElement就是通过vnode生成真实DOM挂载到对应的节点上。

function mountElement(vnode, container) {

    if (typeof vnode === "string") { //字符串节点
        container.appendChild(document.createTextNode(vnode)); // 创建文本节点直接挂载
        return;
    }

    const el = document.createElement(vnode.tagName);
    const {
        props
    } = vnode;
    for (const key in props) {
        if (Object.hasOwnProperty.call(props, key)) {
            if (key.startsWith("on")) {
                // 事件属性
                const eventType = key.substring(2).toLowerCase();
                el.addEventListener(eventType, props[key]);
            } else {
                el[key] = props[key];
            }
        }
    }
    const {
        children
    } = vnode;
    if (typeof children === "string") {
        // 子节点是个字符串节点
        el.appendChild(document.createTextNode(children));
    } else if (Array.isArray(children)) {
        // 子节点是一个子节点数组时,递归创建节点并挂载到当前节点
        children.forEach(node => mountElemnet(node, el));
    }

    container.appendChild(el);
}

为了后续进行统一处理,我们封装一个渲染器函数renderer

function renderer(vnode, container) {
    // 一个普通DOM元素节点
    mountElement(vnode, container);
}

此时已经可以处理普通的虚拟DOM。

const vnode = createVnode("div", {
    onClick: () => {
        console.log(123);
    }
},[
    "123",
    createVnode("img", {
        src : "https://img0.baidu.com/it/u=1628473271,2485762845&fm=253&fmt=auto&app=138&f=JPEG?w=500&h=695",
        alt : '这是一张图片'
    }),

])

renderer(vnode,document.querySelector("#app"));

那如果是组件对象怎么办呢?

对于组件对象,我们这么描述

const myComponent = {
    tagName: "Component",
    render() {
        const a =  h("div", {
            onClick: () => {
                console.log(123);
            }
        }, [
            "123",
            h("img", {
                src: "https://img0.baidu.com/it/u=1628473271,2485762845&fm=253&fmt=auto&app=138&f=JPEG?w=500&h=695",
                alt: '这是一张图片'
            }),

        ])
        console.log(a);
        return a;
    }
}

是不是跟我们平常写的组件有点类似

那么其实相比较虚拟DOM,其实就是多一个步骤,调用组件对象的render函数生成虚拟DOM树。

function mountComponent(comp, container) {
    // 用来挂载组件对象
    const subtree = comp.render(); // 调用render函数生成虚拟DOM树
    renderer(subtree, container);
}

/**
 * 
 * @param {Vnode} vnode 虚拟DOM树
 * @param {DOM} container 挂载的DOM 
 */
export function renderer(vnode, container) {
    if (vnode.tagName === "Component") {
        // 说明是一个组件节点
        mountComponent(vnode, container);
    } else if (typeof vnode.tagName === "string") {
        // 说明是一个普通DOM元素节点
        mountElement(vnode, container);
    }
}

那么这样的一个渲染器就可以渲染组件对象了。

const myComponent = {
    tagName: "Component",
    render() {
        const a =  h("div", {
            onClick: () => {
                console.log(123);
            }
        }, [
            "123",
            h("img", {
                src: "https://img0.baidu.com/it/u=1628473271,2485762845&fm=253&fmt=auto&app=138&f=JPEG?w=500&h=695",
                alt: '这是一张图片'
            }),

        ])
        console.log(a);
        return a;
    }
}

renderer(myComponent, document.querySelector("#app"))

通过一个简单渲染器大概知道了渲染器是做什么的。

当然通过h函数写虚拟DOM,我们平常很少这么做,比较麻烦。

而是通过类html的语法写template,而我们写的模板,最后都会通过一个模板编译模块生成render函数。

所以我们开发能够这么方便,要多亏Compiler这个模板编译器。

这里还涉及到两种编译状态,一个是运行时编译,这个不常见,就是Compiler将模板编译成render函数是发生在页面创建的时候,这个时候会消耗一定的性能,且可能会带来页面短暂白屏的问题。

另一个就是预编译,我们一般用的都是这个,就是我们用脚手架搭建的项目,最后打包结果里一般是没有Compiler这个模块的代码的,且也不会有模板代码,模板代码已经编译成render函数。

使用虚拟DOM既然是为了diff算法服务的,那么为什么不用真实DOM进行对比呢?

这里我说一下我的理解,一方面虚拟DOM属于JS层面的,真实DOM属于渲染引擎层面的,js执行线程直接操作js对象一定要比操作真实DOM要快。我们也都知道直接操作真实DOM的代价是比较昂贵的,所以虚拟DOM + diff 的目的就是让我们尽可能少的操作真实DOM。

另一方面,虚拟DOM描述的属性比较少,当然这一点我感觉没什么,因为Vue3的patchflag可以解决这个问题,还有一个问题,我觉得应该就是重排重绘的问题,我们都知道访问一些DOM信息的api时,浏览器为了获取实时的信息,会强制性重排重绘。那么如果用的真实DOM进行diff的话,不免会涉及到offsetWidth等一些属性的比对,这我觉得应该也是虚拟DOM的一个优势。

好了大概就讲这么多了,东西虽然比较浅,但是应该也能有所收获吧。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/38932.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【雷达通信】雷达探测项目仿真附Matlab代码

✅作者简介&#xff1a;热爱科研的Matlab仿真开发者&#xff0c;修心和技术同步精进&#xff0c;matlab项目合作可私信。 &#x1f34e;个人主页&#xff1a;Matlab科研工作室 &#x1f34a;个人信条&#xff1a;格物致知。 更多Matlab仿真内容点击&#x1f447; 智能优化算法 …

《恋上数据结构与算法》第1季:链表原理实现(图文并茂)

数据结构与算法的学习笔记目录&#xff1a;《恋上数据结构与算法》的学习笔记 目录索引链表原理实现一、链表二、链表的设计三、链表的接口设计四、链表接口的实现1. 索引越界的判断2. 根据索引查找指定节点3. 添加数据4. 插入元素5. 删除元素6. 清空元素7. 修改元素8. 查找元素…

傻白入门芯片设计,RDL/Interposer/EMIB/TSV(三)

目录 一、再分配层&#xff08;RDL&#xff09; 二、硅中介层&#xff08;Si Interposer&#xff09;&#xff1a;Active and Passive 三、嵌入式硅桥&#xff08;EMIB&#xff09; 四、硅通孔 TSV&#xff08;Through Silicon Vias&#xff09; 一、再分配层&#xff08;R…

CCF CSP认证2022年6月 归一化处理、寻宝!大冒险!、光线追踪

这是我第一次参加了这次CSP考试&#xff0c;300分&#xff0c;写了124三题&#xff0c;模拟题到现在都没看过题面没看&#xff0c;笑&#xff0c;t4写成模拟加数据结构&#xff0c;200行&#xff0c;因为一个小错误调了1h&#xff0c;错失了大好机会。考试环境的VSC配置的字体太…

[一篇读懂]C语言十讲:单链表的新建、查找

[一篇读懂]C语言十讲&#xff1a;单链表的新建、查找1. 与408关联解析及本节内容介绍1 与408关联解析2 本节内容介绍2. 头插法新建链表实战3. 尾插法新建链表实战4. 按位置查找及按值查找实战5. 往第i个位置插入元素实战6. 链表的调试方法总结234561. 与408关联解析及本节内容介…

面对无法投入模型训练的object类型数据在头疼,快来使用我的丝滑小连招

面对无法投入模型训练的object类型数据在头疼&#xff0c;快来使用我的丝滑小连招 前言 丝滑小连招 tip1- get_dummies完美one-hot&#xff08;str->int&#xff09; tip2 - rename_dims解决重名问题&#xff01; tip3 - insert且drop&#xff01;​​​​​​​ 前言 我…

小爱同学控制美的美居中的家电热水器,空调等

背景 家里大多数家电都是支持接入米家App的&#xff0c;美的家电不能接入小米&#xff0c;电脑安装Home Assistant成功实现小爱语音控制美的燃气热水器。 实现步骤&#xff1a; 1. 安装docker 我的电脑是windows的&#xff0c;那就直接安装docker desktop https://desktop.…

【Linux】基础指令(三) —— 收尾篇

文章目录前言zip 和 unzip 指令tar 指令bc 指令uname 指令history关机热键补充ctrl c↑ && ↓ctrl rctrl d指令拓展结语前言 今天为大家带来的是最后一部分基础指令讲解。主要内容为 7个指令讲解、热键补充、简单提一下指令的拓展 。内容相对之前较少&#xff0c;更…

服务器密码以及用户名怎么修改

服务器密码以及用户名怎么修改 我是艾西&#xff0c;今天给大家说下服务器密码如何修改 windows2003系统&#xff1a; 1、右键我的电脑&#xff0c;点击“管理”&#xff1a; 2、在“本地用户和组”中打开“用户”&#xff0c;在右侧找到 Administrator 账户进行修改。 200…

【linux】linux实操篇之任务调度

目录前言crond 任务调度概述基本语法快速入门案例案例一&#xff1a;每隔一分钟将ls -l /etc/ 追加到 /tmp/to.txt 文件案例二&#xff1a;每隔一分钟执行python文件结语前言 我们常用linux做一些定时任务&#xff0c;最常见的就是在服务器领域&#xff0c;我们常常做一些定时…

高分辨率格式理论

一个核心概念&#xff1a;人工粘性 考虑经典的双曲守恒律方程 ∂u∂t∂f∂x0{{\partial u} \over {\partial t}} {{\partial f} \over {\partial x}} 0∂t∂u​∂x∂f​0 可以写成守恒形式的数值格式 uin1uin−λ(f^i1/2n−f^i1/2n)u_i^{n 1} u_i^n - \lambda \left( {\ha…

基于ssm+mysql+jsp学生成绩管理系统(含实训报告)

基于ssmmysqljsp学生成绩管理系统(实训报告&#xff09;一、系统介绍二、功能展示1.学生信息查询2.学生信息添加3.学生信息修改4.学生信息删除四、获取源码一、系统介绍 系统主要功能&#xff1a;系统实现了学生信息查询、添加、修改、删除。 环境配置&#xff1a; Jdk1.8 M…

[附源码]java毕业设计智慧教学平台

项目运行 环境配置&#xff1a; Jdk1.8 Tomcat7.0 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&#xff1a; SSM mybatis Maven Vue 等等组成&#xff0c;B/S模式 M…

储能辅助电力系统调峰的容量需求优化配置matlab/cplex

参考文献&#xff1a;储能辅助电力系统调峰的容量需求研究 摘要&#xff1a;建立了储能辅助电力系统调峰的容量需求优化配置模型&#xff0c;设置了含储能和不含储能两种仿真方案&#xff0c;将两个算例代入所提模型进行求解&#xff0c;得到最优的储能系统容量和功率配置&…

Flutter高仿微信-第52篇-群聊-清空聊天记录

Flutter高仿微信系列共59篇&#xff0c;从Flutter客户端、Kotlin客户端、Web服务器、数据库表结构、Xmpp即时通讯服务器、视频通话服务器、腾讯云服务器全面讲解。 详情请查看 效果图&#xff1a; 实现代码&#xff1a; //清空聊天记录对话框 void _cleanGroupChatDialog(){Lo…

【Python】数据类型 + 运算符 + 输入输出

文章目录一. 常量和表达式二. 变量和类型1. 什么是变量2. 变量的语法2.1 定义变量2.2 使用变量3. 变量的类型3.1 整数3.2 浮点数3.3 字符串3.4 布尔3.5 关于变量类型的几点补充三. 注释1. 什么是注释&#xff1f;2. 为什么要有注释&#xff1f;3. 如何写注释&#xff1f;3.1 注…

深度学习制作自己的数据集—为数据集打上标签保存为txt文件,并进行划分和加载数据集

目录 0 前言 1 为图片数据集打上标签并保存为txt文件 2 将txt文件中的图片标签数据集随机划分为训练集和测试集 3 加载txt文件中的图片标签数据集 0 前言 目前是被封控的第四天了&#xff0c;只能呆在宿舍不能出去&#xff0c;记得上次这样子还是一年前大四快毕业那时候了……

CyberController手机外挂番外篇:源代码的二次修改

文章目录前言调试过程中的疑问为什么一段时间不使用CyberController&#xff0c;翻译就无法触发了&#xff1f;为什么连接成功了&#xff0c;但却依然无法进行语音识别和翻译&#xff1f;多长时间TCP连接就会挂掉连接正常与断开连接有什么区别&#xff1f;不停进行翻译&#xf…

现代密码学导论-18-伪随机置换

目录 伪随机置换 PROPOSITION 3.26 伪随机置换和伪随机函数的关系 DEFINITION 3.27 强伪随机置换 伪随机置换 我们称F是含参数k的置换&#xff0c;当且仅当 且对于所有k&#xff0c; Fk是一对一的&#xff0c;即是满射的。 其中 lin 称为F的块长度 对于给定的 k、x和k、y&…

76.【图】

图( 一).图的基本结构(1).无序偶对.(2).有序偶对(3).有向图和无向图(4).权(5).网图(二).图的基本术语(1).邻接.依附(2).有向完全图,无向完全图(3).顶点的度,入度,出度(4).路径 路径长度 回路(5).简单路径 简单回路(6).子图(7).连通图 连通分量(8).强连通图 强连通分量(三).图的…