浅识vue的虚拟DOM和渲染器

news/2024/5/18 11:55:52/文章来源:https://blog.csdn.net/weixin_45696837/article/details/128053944

虚拟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<imgsrc="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.luyixian.cn/news_show_226065.aspx

如若内容造成侵权/违法违规/事实不符,请联系dt猫网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【面试】揭秘面试背后的那点真实

注&#xff1a;最后有面试挑战&#xff0c;看看自己掌握了吗 文章目录前言/背景面试流程资料总结/刷题指南个人经验总结寄语&#x1f338;I could be bounded in a nutshell and count myself a king of infinite space. 特别鸣谢&#xff1a;木芯工作室 、Ivan from Russia 金…

QT:debug日志—打不开头文件以及qDebug和Q_CLASSINFO的使用

这个是因为链接器在给定路径上搜索不到对应的头文件&#xff0c;而大多数的Qt相关的头文件都集中在一个include文件夹里&#xff1a; 我电脑上的路径是&#xff1a;C:\Qt\Qt5.9.7\5.9.7\msvc2017_64\include 然后我们在项目设置里&#xff1a; 注意&#xff0c;这边要加上\*&…

计算机系统基础期末复习

C语言代码如下&#xff1a; void fun(int n){ int x n*12;int y n/32; }请将其中计算的部分优化为位运算、移位运算和加法运算的结合。 x n8n4 (n<<3)(n<<2) x (n(n>>31) & 0x1F)>>5 设32位的位串为x(x类型为unsigned int)&#xff0c;现要…

Flink常用Sink(elasticsearch(es)Sink、RedisSink、KafkaSink、MysqlSink、FileSink)

flink输出到es、redis、mysql、kafka、file 文章目录配置pom文件公共实体类KafkaSInkElasticsearchSink(EsSink)RedisSinkMysqlSink(JdbcSink)FileSink自己先准备一下相关环境 配置pom文件 <properties><maven.compiler.source>8</maven.compiler.source>&l…

测试用例设计方法之场景设计法

基本流&#xff1a;采用直黑线表示&#xff0c;是经过用例的最简单的路径&#xff08;无任何差错&#xff0c;程序从开始直接执行到结束&#xff09; 备选流&#xff1a;采用不同颜色表示&#xff0c;一个备选流可能从基本流开始&#xff0c;在某个特定条件下执行&#xff0c;…

HTTP介绍报文格式构造

HTTP 一. 简单介绍一下: 二. 学习报文格式: 三. HTTP中的细节介绍 四, 如何构造一个HTTP请求 一. 简单介绍一下: 是应用层的典型协议客户端发送一个HTTP请求, 服务器返回一个HTTP响应(一问(请求)一答(响应)的)HTTP是文本格式的协议二. 学习报文格式: 1)先简单看一看HTTP的…

在CentOS 7.7 x86_64上为python 2.7.5安装pip的靠谱方法

我的虚拟机是CentOS 7.7 x86_64系统&#xff0c;对应的python默认版本是2.7.5&#xff0c;但是没有安装pip&#xff0c;不方便安装第三方模块。 我想为为它安装pip工具&#xff0c;发现现有的安装方法都行不通了&#xff0c;比如先安装easy_install&#xff0c;再通过easy_inst…

Nginx (4):nginx动静分离

什么是动静分离不解释了&#xff0c;网上说的很清楚&#xff0c;这里只说配置 目的 02虚拟机运行一个tomcat&#xff0c;处理动态请求&#xff0c;而对静态文件的访问则交给01虚拟机。操作 下面是01虚拟机的配置文件内容&#xff1a; server {listen 82;listen [::]:82;#root /…

pytorch案例代码-3

双向循环神经网络 双向循环神经网络在RNN/LSTM/GRU里都有。比如RNN cell&#xff0c;只是把h0和x1传入做线性变换产生h1继续传入同一个cell做线性变换&#xff0c;线性变换的W和b共享&#xff0c;沿着这个方向就把所有隐层和最后的输出算出来了。 那么其中的每个结点&#xff0…

文华财经期货K线多周期画线技术,多重短线技术共振通道线指标公式——多周期主图自动画线

期货指标公式是通过数学逻辑角度计算而来&#xff0c;仅是期货分析环节中的一个辅助工具。期货市场具有不确定性和不可预测性的&#xff0c;请正常对待和使用指标公式! 期货指标公式信号本身就有滞后性&#xff0c;周期越大&#xff0c;滞后性越久。指标公式不是100%稳赚的工具…

18.4 嵌入式指针概念及范例、内存池改进版

一&#xff1a;嵌入式指针&#xff08;embedded pointer&#xff09; 1、嵌入式指针概念 一般应用在内存池相关的代码中&#xff0c;成功使用嵌入式指针有个前提条件&#xff1a;&#xff08;类A对象的sizeof必须不小于4字节&#xff09; 嵌入式指针工作原理&#xff1a;借用…

Word2Vec 实践

Word2Vec 实践 gensim库使用 这里的Word2Vec借助 gensim 库实现&#xff0c;首先安装pip install gensim3.8.3 from gensim.models.word2vec import Word2Vecmodel Word2Vec(sentencesNone, size100, alpha0.025, window5, min_count5,max_vocab_sizeNone, sample1e-3, …

2023年系统规划与设计管理师-第三章信息技术服务知识

一. 思维导图 二.IT 服务管理 (ITSM) 1. 什么是 IT 服务管理 (ITSM)&#xff1f; IT 服务管理 (ITSM) 包含一组策略和实践&#xff0c;这些策略和实践可用于为最终用户实施、交付和管理 IT 服务&#xff0c;以满足最终用户的既定需求和企业的既定目标。 在此定义中&#xff0…

cocos2dx创建工程并在androidstudio平台编译

本文主要是通过androidstudio进行编译运行cocos2dx工程。 前置条件&#xff1a; 1&#xff1a;androidstudio已经下载并安装。 2&#xff1a;cocos2dx已经下载并打开。 这里androidstudio使用2021.3.1版本&#xff0c;cocos2dx使用4.0版本。 第一步&#xff0c;首先安装py…

基于JavaWeb的药品进销存管理系统(JSP)

目 录 绪论 1 1.1 本课题的研究背景 1 1.2 国内外研究现状 1 1.3 本课题的主要工作 2 1.4 目的和意义 2 开发工具及技术 3 2.1 开发工具 3 2.1.1 MyEclipse 3 2.1.2 Tomcat 3 2.1.3 Mysql 3 2.2 开发技术 4 2.2.1 JSP 4 2.2.2 MyBatis 4 2.2.3 JavaScript 4 2.2.4 jQuery以及j…

六、nacos环境隔离、服务配置拉取和多环境配置共享

文章目录一、环境隔离-namespace1.namespace理解2.创建命名空间二、Nacos-实现配置管理三、nacos-实现服务配置拉取1.非热更新2.热更新&#xff1a;四、实现多环境配置共享1.开发环境&#xff1a;2.测试环境3.结论一、环境隔离-namespace 1.namespace理解 Nacos中服务存储和数…

Servlet到底是什么(非常透彻)

Servlet到底是什么&#xff1f;1. Servlet的概念2. Servlet是一种规范3. Servlet的接口4. JSP是什么学习顺序1. Servlet的概念 Servlet 是 Server Applet 的缩写&#xff0c;译为“服务器端小程序”&#xff0c;是一种使用 Java 语言来开发动态网站的技术。 Servlet 虽然被称…

开发工具vim

一、开发工具vim vim的安装&#xff1a;yum install -y vim 指令&#xff1a;vim --version可以查看当前的版本信息&#xff0c;没有弹出信息就是没有安装。 装的vim是没有相关配置文件的&#xff0c;配置文件需要后面自己装。 之前说过&#xff0c;vs2019是集成软件编译…

Go学习之旅:包、变量和函数(DAY 1)

文章目录前引包、变量和函数1、包的概念和所用2、导出名或者导出函数3.1、函数参数声明方式&#xff08;一&#xff09;3.2、函数参数声明方式&#xff08;二&#xff09;4、函数返回值支持多值返回5、函数命名返回值6、变量声明7、变量的基础类型8、变量的默认值&#xff08;零…

Qt5开发从入门到精通——第十二篇二节(Qt5 事件处理及实例——多线程控制、互斥量、信号量、线程等待与唤醒)

提示&#xff1a;欢迎小伙伴的点评✨✨&#xff0c;相互学习c/c应用开发。&#x1f373;&#x1f373;&#x1f373; 博主&#x1f9d1;&#x1f9d1; 本着开源的精神交流Qt开发的经验、将持续更新续章&#xff0c;为社区贡献博主自身的开源精神&#x1f469;‍&#x1f680; 文…