vue如何通过VNode渲染节点

news2025/8/8 6:48:41

vue如何通过VNode渲染节点

    • vue的源码包含三大核心
    • 实现一个Mini-Vue
      • 渲染系统的实现
    • vue2和vue3写法上的区别

vue的源码包含三大核心

Compiler模块:编译模板系统

Runtime模块:也可以称之为Renderer模块,真正的渲染的模块

Reactivity模块:响应式系统

三个系统之间如何协同工作呢?
在这里插入图片描述

实现一个Mini-Vue

包含三个模块:

渲染系统模块

可响应式系统模块

应用程序入口模块

渲染系统的实现

该模块主要包含三个功能:

功能一:h函数,用于返回一个VNode对象;

功能二:mount函数,用于将VNode挂载到DOM

功能三:patch函数,用于对两个VNode进行对比,决定如何处理新的VNode
第一步,创建一个renderer.js文件,定义一个h函数

const h = (tag, props, children) => {
    // vnode就是一个JavaScript对象
    return {
        tag,
        props,
        children
    }
}

在html文件中,引入文件,并创建一个虚拟节点,可以输出打印一下这个vnode

<div id="app"></div>
        <script src="./renderer.js"></script>
        <script>
            // 1、 通过h函数来创建一个vnode
            const vnode = h('div', {
                class: 'vnode'
            }, [
                h('h2', null, '当前计数:100'),
                h('button', null, "+1")
            ])
            console.log(vnode)
        </script>

在这里插入图片描述
第二步,实现挂载功能
在renderer.js文件中定义mount方法

const mount = (vnode, container) => {
    //1、 将vnode变为elemnt,创建出真实的dom,并且在vnode上保存一份el
    const el = vnode.el = document.createElement(vnode.tag)
    //    2、处理props
    if (vnode.props) {
        for (const key in vnode.props) {
            const value = vnode.props[key]
            // 判断传递过来的是否是方法,比如onClick
            if (key.startsWith("on")) {
                el.addEventListener(key.slice(2).toLowerCase(), value)
            }
            // 设置属性
            el.setAttribute(key, value)
        }
    }
    // 3、处理children
    if (vnode.children) {
        // 如果子节点存在并且子节点是字符串,说明是其中的内容
        if (typeof vnode.children === 'string') {
            // 将内容放进去
            el.textContent = vnode.children
        } else {
            // 说明子节点中是一个数组,其内部还有子节点
            vnode.children.forEach((item) => {
                // 再次调用挂载到el上
                mount(item,el)
            })
        }
    }
    // 4、将el挂载到container上
    container.appendChild(el)
}

在html文件中调用该mount方法

// 2、通过mount函数,将vnode挂载到#app上
mount(vnode,document.getElementById('app'))

再次刷新页面的时候就可以看到界面已经加载出来了vnode
在这里插入图片描述
第三步实现diff算法
第一种情况:节点不相同
新建一个vnode

 // 3、创建一个新的vnode
            const vnode1 = h('h2', {
                class: 'vnode'
            }, 'jerry')

将新的vnode替换旧的vnode,两个vnode之间进行一个diff算法,根据diff算法找到需要修改真实dom的那个地方,找到之后在进行修改

在renderer.js文件中定义一个patch方法

const patch=(n1,n2)=>{
    // 判断两个vnode的类型是否一样,比如说n1为div,n2为h2
    if(n1.tag!==n2.tag){
        // 拿到n1节点的父元素
        const n1ElementParent=n1.el.parentElement;
        // 移除n1节点
        n1ElementParent.removeChild(n1.el)
        // 将n2节点添加上去
        mount(n2,n1ElementParent)
    }else{

    }
}

在html文件中使用patch方法

patch(vnode,vnode1)

再次刷新页面可以看到已经替换
在这里插入图片描述
第二种情况:节点相同,类名不同

patch方法

const patch = (n1, n2) => {
    // 判断两个vnode的类型是否一样,比如说n1为div,n2为h2
    if (n1.tag !== n2.tag) {
        // 拿到n1节点的父元素
        const n1ElementParent = n1.el.parentElement;
        // 移除n1节点
        n1ElementParent.removeChild(n1.el)
        // 将n2节点添加上去
        mount(n2, n1ElementParent)
    } else {
        // 1、拿出element对象,并在n2中保留一份
        const el = n2.el = n1.el
        // 2、处理props
        const oldProps = n1.props || {}
        const newProps = n2.props || {}
        // 2、1获取所有的newProps添加到el中
        for (const key in newProps) {
            const oldValue = oldProps[key]
            const newValue = newProps[key]
            if (newValue !== oldValue) {
                // 判断传递过来的是否是方法,比如onClick
                if (key.startsWith("on")) {
                    el.addEventListener(key.slice(2).toLowerCase(), newValue)
                } else {
                    el.setAttribute(key, newValue)
                }
            }
        }
        // 2、2删除旧的props
        for(const key in oldProps){
            if(!(key in  newProps)){
                if (key.startsWith("on")) {
                    const value=oldProps[key]
                    el.removeEventListener(key.slice(2).toLowerCase(), value)
                } else {
                    el.removeAttribute(key)
                }
            }
        }
        // 3、处理children
    }
}

在html中新建一个节点,调用patch方法

 // 3、创建一个新的vnode
            const vnode1 = h('div', {
                class: 'jerry'
            }, 'jerry')
            patch(vnode,vnode1)

之前
在这里插入图片描述
更新之后
在这里插入图片描述
接下来处理子节点

// 3、处理children
        const oldChildren = n1.children || [];
        const newChildren = n2.children || [];
        // 情况一:newChildren是一个string类型
        if (typeof newChildren === "string") {
            if (typeof oldChildren === "string") {
                if (newChildren !== oldChildren) {
                    el.textContent = newChildren
                }
            } else {
                el.innerHTML = newChildren;
            }
        }else{
            // 情况二:newChildren是一个数组
            if(typeof oldChildren==='string'){
                el.innerHTML=""
                newChildren.forEach(item=>{
                    mount(item,el)
                })
            }else{
                // oldChildren:[n1,n2,n3]
                // newChildren:[n1,n2,n3,n4,n5]
                // 前面有相同节点的元素进行patch操作
                const commonLength=Math.min(oldChildren.length,newChildren.length)
                for(let i=0;i<commonLength;i++){
                    patch(oldChildren[i],newChildren[i])
                }
                // 如果newChildren.length>oldChildren
                // oldChildren:[n1,n2,n3]
                // newChildren:[n1,n2,n3,n4,n5]
                if(newChildren.length>oldChildren.length){
                    newChildren.slice(oldChildren.length).forEach(item=>{
                        mount(item,el)
                    })
                }
                // 如果newChildren.length<oldChildren
                // oldChildren:[n1,n2,n3,n4,n5]
                // newChildren:[n1,n2,n3]
                if(newChildren.length<oldChildren.length){
                   oldChildren.slice(newChildren.length).forEach(item=>{
                       el.removeChild(item.el)
                   })
                }
            }
        }
    }

创建两个不同的节点,在进行patch操作

// 1、 通过h函数来创建一个vnode
            const vnode = h('div', {
                class: 'vnode'
            }, [
                h('h2', null, '当前计数:100'),
                h('button',{onClick:function(){}}, "+1")
            ])
            
            // 2、通过mount函数,将vnode挂载到#app上
            mount(vnode,document.getElementById('app'))
            // 3、创建一个新的vnode
            const vnode1 = h('div', {
                class: 'jerry'
            }, 'jerry')
            patch(vnode,vnode1)

此时页面就成为:
在这里插入图片描述

vue2和vue3写法上的区别

主要是在获取h函数以及事件绑定上有区别
vue2

const h = this.$createElement;

const vnode = h('div', {
	class: 'v-node-ele',
	on: {
		click: () => {
			console.log('点击事件')
		}
	}
}, '虚拟节点内容')

vue3

import { h } from 'vue';

const vnode = h('div', {
	class: 'v-node-ele',
	onClick: () => {
	  console.log('点击事件')
	}
}, h(
	'span', null, 'children内容'
))

参考:
https://www.cnblogs.com/keyeking/p/16112165.html
https://blog.csdn.net/txf666/article/details/124755693
https://blog.csdn.net/qq_42009005/article/details/122986362

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

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

相关文章

合成孔径雷达地面运动目标检测技术研究——基于概率图(Matlab代码实现)

&#x1f352;&#x1f352;&#x1f352;欢迎关注&#x1f308;&#x1f308;&#x1f308; &#x1f4dd;个人主页&#xff1a;我爱Matlab &#x1f44d;点赞➕评论➕收藏 养成习惯&#xff08;一键三连&#xff09;&#x1f33b;&#x1f33b;&#x1f33b; &#x1f34c;希…

SpringCloud学习笔记(四)

文章目录SpringCloud学习笔记(四)1.说在前面2.OpenFeign 简介3.OpenFeign 快速入门3.1 本次调用的设计图3.2 启动一个 eureka-server 服务&#xff0c;这里不重复演示&#xff0c;参考 eureka3.3 先创建 01-order-service&#xff0c;选择依赖3.4 创建 02-user-consumer&#x…

Uni-app常用知识点总结

一、一句话总的形容一下uniapp与vue和微信小程序的异同点 简单来讲Uni-app就是用着vue的指令和小程序的组件和API 二、Uniapp中配置tabbar的方式 见之前的单独文章—— (3条消息) Uni-app中的tabBar的配置_终将抵达丶的博客-CSDN博客_uniapp设置tabbar图片大小https://blog.…

利用霍尔效应传感器和Arduino研究了一个简单的钟摆

A simple pendulum studied using Hall effect sensor and Arduino 利用霍尔效应传感器和Arduino研究了一个简单的钟摆&#xff1a;原文&#xff08;Hall effect sensor (scitation.org)&#xff09; ARTICLES YOU MAY BE INTERESTED IN Measurement of gravitational accele…

One UI 5 升级来了

从11月23日开始&#xff0c;三星多款手机海内外开始推送安卓13/One UI 5.0正式版&#xff0c;大家心心念念的One UI 5终于来了&#xff0c;接下来我们看下有关新版One UI 5相关的更新内容&#xff0c;具体如下&#xff1a; One UI 5 升级 (Android 13) One UI 5 为您带来更加强…

Eureka服务注册与发现

✨ Eureka服务注册与发现微服务的注册中心注册中心的基本介绍注册中心的主要作用注册中心基本原理常见的注册中心Eureka基本介绍服务治理服务注册Eureka 两大组件搭建EurekaEureka端服务注册中心创建新模块 cloud-eureka-server7001添加pom依赖yml配置启动类服务中心管理后台服…

Connection(数据库连接对象)

Connection&#xff08;数据库连接对象&#xff09; 简介&#xff1a;通过代码来讲解Connection的含义。 推荐学习路线&#xff1a;JDBC数据库的连接->Connection&#xff08;数据库连接对象&#xff09;->Statement->ResultSet->通过PreparedStatement预防SQL注入…

【云原生 | Kubernetes 系列】--Gitops持续交付 Argo Rollouts Analysis

1. Argo Rollouts 由一个控制器和一组CRD组成,可为K8s提供高级部署功能 - blue-green - canary - canary analysis 结合外部指标系统金丝雀 - experimentation 实验性的结果 - progressive delivery 渐进式交付,精准管控外部流量策略,不用关心后端部署机制支持Ingress Contro…

第六章《类的高级特性》第1节:static关键字的使用

static意为“静态”,在Java语言中,使用static关键字可以定义静态属性、静态方法和静态块。 6.1.1 静态属性 在第5章中,我们定义了一个Person类的子类Student,用它来表示学生。假如每一个在读学生每年都能得到1000元的助学津贴,并且程序员希望在Student类中以属性的形式把…

骨传导有没有副作用?骨传导耳机有什么优点吗?

骨传导有没有副作用&#xff1f; 先说结论&#xff1a;是没有的。 骨传导耳机虽然是近两年在走向我们大众视野&#xff0c;但是骨传导技术早就已经在医疗、军事领域广泛应用&#xff0c;骨传导也不是什么高端的技术&#xff0c;像我们平常嗑瓜子&#xff0c;吃薯片&#xff0…

javaee实验,SSM整合开发综合实例

由于不能使用maven管理&#xff0c;只能导入jar包做实验&#xff0c;最下面有截图展示所用到的jar包&#xff0c;可以自己搜索文档maven导入依赖&#xff1b; SSM整合开发综合实例 实验目的&#xff1a; &#xff08;1&#xff09;掌握SSM项目整合的原则&#xff1b; &#x…

Microsoft Visual Studio C++开发环境的配置及使用

Microsoft Visual Studio C开发环境的配置及使用 本文引用自作者编写的下述图书; 本文允许以个人学习、教学等目的引用、讲授或转载&#xff0c;但需要注明原作者"海洋饼干叔 叔"&#xff1b;本文不允许以纸质及电子出版为目的进行抄摘或改编。 1.《Python编程基础及…

大数据面试题(四):Yarn核心高频面试题

文章目录 Yarn核心高频面试题 一、简述Hadoop1与Hadoop2的架构异同 二、为什么会产生yarn&#xff0c;它解决了什么问题&#xff0c;有什么优势&#xff1f; 三、HDFS的数据压缩算法&#xff1f;及每种算法的应用场景&#xff1f; 1、gzip压缩 2、Bzip2压缩 3、Lzo压缩 …

SpringBoot框架接收参数的六种常用方式(全面详细)

文章目录[toc]一、基于PathVariable注解二、基于RequestParam注解三、基于PathVariableRequestParam混合四、基于RequestBody注解五、基于HttpServletRequest请求六、不基于任何注解进行表单传参一、基于PathVariable注解 RestControllerpublic class UserController {GetMapp…

C#11新特性之原始字符串

随着.NET 7与C#11的发布&#xff0c;微软发布了C# 11 中的原始字符串这个新特性。 这个新特性解决了祖传字符串中引号的问题。 微软官方的表述是:" Raw string literals"&#xff0c;圈里都叫他”原始字符串”。从字面不难看出&#xff0c;它是适用于字符串的新特性…

elasticsearch 之 histogram 直方图聚合

1. 简介 直方图聚合是一种基于多桶值聚合&#xff0c;可从文档中提取的数值或数值范围值来进行聚合。它可以对参与聚合的值来动态的生成固定大小的桶。 2. bucket_key如何计算 假设我们有一个值是32&#xff0c;并且桶的大小是5&#xff0c;那么32四舍五入后变成30&#xff…

两种方式实现css取消页面鼠标双击选中文字或单击拖动选中文字的效果

问题描述 我们知道浏览器页面上的文字正常情况下我们是可以双击选中、或者单击鼠标横向拖动也能选中的&#xff0c;选中以后可以右击出现面板然后去复制什么的。但是有的时候&#xff0c;这种效果我们并不想要的&#xff0c;比如用户点快了的时候&#xff0c;所以我们需要禁用…

TensorFlow之文本分类算法-3

1 前言 2 收集数据 3 探索数据 4 选择模型 5 准备数据 N-gram向量集 序列向量集 序列向量集主要是用于序列模型中对文本执行分词与向量化&#xff0c;与n-gram向量集类似&#xff0c;也使用特征选择与标准化的技术优化序列向量集的表示。 在一些文本样例中&#xff0c;…

GaussDB-物理、逻辑备份 使用方法和[GAUSS-53403]解决办法

文章目录1.逻辑备份-gs_dump2.逻辑备份恢复数据库3.物理备份&#xff08;分布式集群验证&#xff09;查看物理全量备份集&#xff1a;查看物理增量备份集&#xff1a;查看所有备份集&#xff08;该命令无法确定备份是否有效&#xff09;停止物理备份&#xff1a;使用物理备份集…

centos7安装mysql8.0.31

mysql 官网 https://www.mysql.com/ 找到对应的版本 然后下载 连接虚拟机 mysql 会和 mariadb这个有冲突&#xff0c;需要卸载掉 查看是否有mariadb rpm -qa|grep mariadb rpm -e --nodeps mariadb-libs 这个是强制卸载命令 再查看一下 rpm -qa|grep mariadb 在根目录创建…