vue —— h函数的学习与使用

文章目录

  • 一、h函数是什么?
  • 二、h函数格式说明及使用
    • 示例1:简单创建一个VNode(vue3)
    • 示例2:vue2中h函数用法
    • 示例3:vue3中h函数的用法
    • vue2和vue3中h函数的区别?
  • 三、h函数实现原理
  • 四、h函数常用场景
  • 五、h函数及插槽slot使用
  • 六、h函数和v-model使用

一、h函数是什么?

先看一下 Vue3官方文档说明:
在这里插入图片描述

简单理解: Vue中的 h 函数是 Vue 用于创建虚拟DOM(Virtual DOM)节点的核心函数。在 Vue 3.0 版本中,这个函数作为渲染函数的基础工具,用于替代模板解析的过程,允许开发者以编程方式描述组件结构和动态生成虚拟节点。

二、h函数格式说明及使用

function h(type, props, children)
  • type: 必需,表示要创建的元素类型或组件类型。它可以是一个字符串(HTML标签名),一个组件选项对象、异步组件函数或者一个函数式组件。
  • props: 可选的对象,包含了传递给元素或组件的所有属性(attributes)和 props。例如:可以包含类名、样式、事件监听器等。
  • children: 可选,代表子节点,它可以是字符串(文本内容)、数组(包含多个子节点,每个子节点可以是字符串或其他由 h 创建的虚拟节点)或者其他合法的虚拟DOM节点。

示例1:简单创建一个VNode(vue3)

import { h } from 'vue'const vnode = h('div', // 类型为 div 标签{ id: 'app', class: 'container' }, // props 对象[ // 子节点数组h('p', { class: 'text' }, 'Hello, World!'), // 子节点1:一个 p 标签h('button', { onClick: handleClick }, 'Click me') // 子节点2:一个绑定了点击事件handleClick的按钮]
)function handleClick() {console.log('Button was clicked!')
}

通过h函数的方式构建虚拟DOM树,Vue可以在内部高效地比较新旧虚拟DOM的不同,并最小化地更新实际DOM,从而提高页面渲染性能。

示例2:vue2中h函数用法

import Vue from 'vue' // 引入Vue和h函数// 定义一个组件
var MyComponent = {props: ['message'],render: function (createElement) {return createElement( // createElement实际上就是h函数'div', // 元素类型{ 		 // 属性对象class: ['my-class'], style: { color: 'red' },on: {click: this.handleClick,},},[ // 子元素数组this.message, // 文本内容createElement('span', {}, '这是一个子元素'),])},methods: {handleClick: function (event) {console.log('Clicked!')}}
}// 注册并使用该组件
new Vue({el: '#app',components: {MyComponent},template: `<div id="app"><my-component message="Hello from custom render function!" /></div>`
})

示例3:vue3中h函数的用法

import { h, ref, defineComponent } from 'vue';// 定义一个使用h函数的组件
const MyComponent = defineComponent({props: {message: String,},setup(props) {const count = ref(0);function handleClick() {count.value++;}return () => h('div', { // 元素类型class: ['my-class'], style: { color: 'red' },onClick: handleClick, // 事件监听器}, [props.message, // 文本内容h('span', {}, '这是一个子元素'),`Clicked ${count.value} times`, // 动态文本内容]);},
});// 使用该组件
new Vue({render: (h) => h(MyComponent, { props: { message: "Hello from custom render function!" } }),
}).$mount('#app');

vue2和vue3中h函数的区别?

1、导入方式不同: Vue 2需要从 Vue 实例中获取 h 函数,通常在 render 函数中通过参数传递。Vue 3直接从 @vue/runtime-domvue 模块中导入。

// vue2
import Vue from 'vue';
export default {render(h) {// 使用 h 函数}
}// vue3
import { h } from 'vue';
export default {setup() {return () => h('div', 'Hello World');}
}

2、与Composition API集成

  • Vue 2不直接支持 Composition API,需要结合 options API 来使用。
  • Vue 3在 Composition API 的 setup 函数中可以直接使用响应式数据,并返回一个渲染函数作为组件的输出。

3、模板语法的替代

  • Vue 2在自定义渲染函数中广泛使用 h 函数来代替模板语法。
  • Vue 3虽然仍然可以使用 h 函数,但随着单文件组件(SFC)中模板语法以及Composition API的改进,直接使用 h 函数的情况相对较少。

4、优化与性能

  • Vue 3底层引擎优化,使得 h 创建的虚拟DOM更加高效,尤其是在大型应用中表现更好。

5、props的处理

  • Vue 3中的 props 是通过 defineComponentprops 属性显式声明的,而在使用 h 函数时,这些 prop 值可以通过 setup 函数中的 props 参数直接访问,无需像 Vue 2 中那样进行解构。

总结来说:Vue 3中的 h 函数保持了其核心作用,但在上下文、API集成度和性能等方面进行了改进和优化,使其更适应于现代Vue应用程序开发需求。

三、h函数实现原理

Vue中的 h 函数(在 Vue 2中也称为 createElement)是用于构建虚拟 DOM 节点的核心函数。

基本实现原理可概述如下:

1、虚拟DOM节点创建:当调用 h(type, props, children) 时,它会创建一个描述真实DOM元素或组件的虚拟对象。这个虚拟对象包含类型信息(如标签名、组件名称)、属性对象(包括类名、样式、事件绑定等)以及子节点列表。

2、响应式数据绑定:如果传递给 h 函数的属性值是响应式的(例如由 Vue 的 datacomputed 返回的值),Vue 将通过内部的响应式系统跟踪这些值的变化,并在需要时重新生成虚拟DOM树。

3、DOM更新优化:Vue 使用虚拟DOM来计算状态变化前后DOM结构的差异,并将最小化的更新应用到实际DOM上,这个过程称为“异步批处理更新”和“DOM diff算法”。当组件状态改变并触发重新渲染时,Vue 会比较新旧虚拟DOM树的差异,只对实际发生变化的部分进行DOM操作,避免了频繁地重绘整个页面,从而提高性能。

4、跨平台支持h 函数作为抽象层,不仅限于浏览器环境下的DOM渲染,还能够应用于服务器端渲染(SSR)以及其他非DOM环境,比如Weex等跨平台应用场景。

5、Vue框架中的应用:Vue 在编译模板阶段,会将模板转换为 render 函数,render 函数中就大量使用了 h 函数来构建虚拟DOM。在Vue 3中即使使用的是模板语法,Vue也会将其编译成基于 h 函数的渲染逻辑。

总结来说,Vue 中的 h 函数通过创建和维护虚拟DOM,结合响应式数据系统与高效的DOM更新策略,实现了高效、灵活的视图渲染机制。

四、h函数常用场景

1、自定义渲染逻辑:当模板语法不足以处理复杂的动态内容或需要根据运行时条件动态生成DOM结构时,可以使用 h 函数在组件的 render 函数中编写更灵活的渲染逻辑。

2、高级组件库开发:组件库开发者通常会依赖 h 函数来创建可复用、功能丰富的组件。通过编程方式构建虚拟节点,可以实现对DOM元素精细控制和优化,如高级表格组件、树形控件、拖拽排序等复杂交互场景。

3、性能优化:在某些性能关键路径上,通过手动编写 h 函数来精确控制DOM更新,避免不必要的子组件渲染,从而提升应用性能。

4、无模板组件:如果不希望或者无法使用 .vue 单文件组件中的 <template> 标签,可以完全基于 h 函数来编写组件的渲染内容。

5、与JSX结合:Vue 3支持 JSX 语法,而 JSX 在编译阶段会被转换为 h 函数调用,因此对于喜欢 React 风格 JSX 开发的开发者来说,h 函数是底层支持的基础。

6、服务端渲染 (SSR):在服务端渲染 Vue 应用时,也需要使用 h 函数来创建 SSR 环境下的虚拟 DOM 节点。

简单使用示例:

// 通过h函数根据传入的items数组动态生成多个<li>子元素,并确保每个子元素都有唯一的key属性,以便Vue进行高效的DOM更新
export default {name: 'CustomComponent',props: ['items'],render() {return h('ul',this.items.map(item => h('li', { key: item.id }, item.text)));}
}

五、h函数及插槽slot使用

import { h, defineComponent } from 'vue';// 定义一个使用插槽的子组件:通过this.$slots来访问并渲染插槽内容
const MyComponent = defineComponent({render() {return h('div', [this.$slots.default?.(), // 使用默认插槽this.$slots.header?.(),  // 使用具名插槽 - 名为"header"this.$slots.footer?.(),  // 使用具名插槽 - 名为"footer"]);},
});// 在父组件中使用MyComponent并插入插槽内容
export default defineComponent({render() {return h(MyComponent, {}, [h('p', '这是默认插槽的内容'), 					// 默认插槽的内容h('template', { slot: 'header' }, [ // 具名插槽"header"的内容h('h1', '这是头部插槽内容'),]),h('template', { slot: 'footer' }, [ // 具名插槽"footer"的内容h('p', '这是底部插槽内容'),]),]);},
});

六、h函数和v-model使用

h函数本身并不直接支持v-model指令,但通过正确设置props和自定义事件,可以模拟出类似的双向数据绑定效果。

示例:使用h函数创建子组件,并实现了类似v-model的功能

// 子组件:实现了v-model的行为
import { h, defineComponent, ref, toRef } from 'vue';const CustomInput = defineComponent({props: {modelValue: { type: String, required: true },},emits: ['update:modelValue'], // 声明将要触发的事件setup(props, { emit }) {const internalValue = ref(props.modelValue); // 创建一个响应式变量保存内部状态function handleChange(event) {internalValue.value = event.target.value; 			// 更新内部状态emit('update:modelValue', internalValue.value); // 触发更新事件}return () => h('input', {type: 'text',value: internalValue.value, // 将内部状态绑定到input元素的value属性onInput: handleChange, 			// 监听输入事件并更新状态});},
});export default CustomInput;
// 父组件
import { h, ref } from 'vue';
import CustomInput from './CustomInput.vue';export default {setup() {const inputValue = ref('初始值');return () => h(CustomInput, {modelValue: inputValue.value,'onUpdate:modelValue': (value) => {inputValue.value = value;},});},
};

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

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

相关文章

《动手学深度学习(PyTorch版)》笔记3.1

Chapter3 Linear Neural Networks 3.1 Linear Regression 3.1.1 Basic Concepts 我们通常使用 n n n来表示数据集中的样本数。对索引为 i i i的样本&#xff0c;其输入表示为 x ( i ) [ x 1 ( i ) , x 2 ( i ) , . . . , x n ( i ) ] ⊤ \mathbf{x}^{(i)} [x_1^{(i)}, x_2…

opencv学习笔记

学习OpenCV3 文章目录 学习OpenCV3openCV模块介绍 图片处理滤波/卷积核高斯滤波 算子索贝尔(sobel)算子沙尔(Scharr)算子拉普拉斯算子Canny边缘检测 **实际使用** 形态学获取形态学卷积核全局二值化自适应阈值二值化腐蚀操作膨胀操作开运算、闭运算形态学梯度顶帽操作黑帽操作 …

Redis 学习笔记 2:Java 客户端

Redis 学习笔记 2&#xff1a;Java 客户端 常见的 Redis Java 客户端有三种&#xff1a; Jedis&#xff0c;优点是API 风格与 Redis 命令命名保持一致&#xff0c;容易上手&#xff0c;缺点是连接实例是线程不安全的&#xff0c;多线程场景需要用线程池来管理连接。Redisson&…

Flink的SQL开发

概叙 Flink有关FlinkSQL的官网: https://nightlies.apache.org/flink/flink-docs-release-1.13/zh/docs/dev/table/sql/overview/ 阿里云有关FlinkSQL的官网: https://help.aliyun.com/zh/flink/developer-reference/overview-5?spma2c4g.11186623.0.0.3f55bbc6H3LVyo Ta…

谷歌seo内容营销怎么做?

谷歌SEO内容营销就像是在为两个观众准备一场表演&#xff0c;一边是用户&#xff0c;一边则是谷歌搜索引擎&#xff0c;那么所谓的内容营销&#xff0c;你自然需要知道你的观众想看什么&#xff0c;这就是关键词研究&#xff0c;帮你了解用户在搜索什么&#xff0c;然后&#x…

表单常用正则表达式(手机,邮箱,身份证,数字,空格...)

#表单常用正式表达式 为了后面项目的拿来即用以便不时之需&#xff0c;特意整理一文&#xff0c;把常用的表单验证的正则表达式整理如下。 不能为空 pattern: /^[^\s]$/ 验证手机号 pattern: /^(13[0-9]|14[01456879]|15[0-35-9]|16[2567]|17[0-8]|18[0-9]|19[0-35-9])\d{8…

【算法专题】动态规划综合篇

动态规划7.0 1. 最长公共子序列2. 不相交的线3. 不同的子序列4. 通配符匹配5. 正则表达式匹配6. 交错字符串7. 两个字符串的最小ASCII删除和8. 最长重复子数组 1. 最长公共子序列 题目链接 -> Leetcode -1143.最长公共子序列 Leetcode -1143.最长公共子序列 题目&#xf…

阿里云购买云服务部署小记

1.先购买云服务器 阿里云网址 阿里云云服务器购买 我这是前端时间出活动买的 点击购买 注: 3M的带宽&#xff0c;可以支持同一秒访问30-35个人不卡(其他依次类似类推) 购买前配置的话 选的也没那么多 一般开发人员根据自己喜好顺手的来 跟我一样是刚接触小白的话 听客服…

微搭低代码从入门到精通01应用介绍

目录 1 学习路线图2 应用介绍3 编辑器介绍总结 低代码的概念于2014年由 Forrester 首次正式提出。其将低代码定义为&#xff1a;能够以“最少的手写代码”和设置快速开发应用、配置和部署业务应用程序。 不同应用厂商的解法不一样&#xff0c;Gartner评估了400多款低代码/无代码…

PWN 常用工具-补充

目录 pwndbg 如何加载文件 如何运行到Main函数停止 如何查看程序的内存布局 查看内存地址存储的值 如何查看某个地址存储的指令 搜索字符串 打印 调试相关 断点相关 查看栈内数据 查看栈调用顺序 从程序回到gdb 修改内存中的值 file 可执行文件特征 动态链接文…

获取鼠标点击图片时候的坐标,以及利用html 中的useMap 和area 实现图片固定位置的点击事件

一 编写原因 应项目要求&#xff0c;需要对图片的固定几个位置分别做一个点击事件&#xff0c;响应不同的操作&#xff0c;如下图&#xff0c;需要点击红色区域&#xff0c;弹出不同的提示框&#xff1a; 二 获取点击图片时候的坐标 1. 说明 实现这以上功能的前提是需要确定需…

【LLVM Pass解读】Reassociate 重结合优化

run函数的分析 首先&#xff0c;ReassociatePass是一个FunctionAnalysis&#xff0c;所以其入口函数为 PreservedAnalyses ReassociatePass::run(Function &F, FunctionAnalysisManager &) { 首先对一个函数的基本块构造ReversePostOrderTraversal&#xff0c;该顺序…