4.6 Vue 3 中的模板引用 (Template Refs)

在 Vue 3 中,ref 是一个核心的响应式 API,但它在模板中还有另一个非常重要的用途:获取对 DOM 元素或子组件实例的直接引用。这就是我们所说的“模板引用”。

核心概念

  • 目的:让你在父组件中能够直接访问并操作特定的 DOM 元素或子组件实例。
  • 场景
    • 手动聚焦一个输入框。
    • 触发一个 DOM 元素上的动画。
    • 调用子组件暴露的特定方法(非响应式数据)。
    • 测量 DOM 元素的尺寸。
    • 与需要直接 DOM 访问的第三方库集成。

基本用法

  1. 在模板中声明 ref

    在你想要引用的元素或组件上,使用 ref attribute(在 Vue 3 的 <script setup> 中,这实际上是一个特殊的指令,但用法像 attribute)。

    <template><!-- 引用一个 DOM 元素 --><input ref="inputRef" type="text" placeholder="请输入..." /><!-- 引用一个子组件 --><ChildComponent ref="childRef" />
    </template>
  2. <script setup> 中定义响应式引用

    使用 ref 函数在 <script setup> 中声明一个变量,这个变量的名字必须与模板中 ref attribute 的值完全一致。Vue 会自动将 DOM 元素或组件实例赋值给这个响应式引用。

    <script setup>
    import { ref, onMounted } from 'vue'
    import ChildComponent from './ChildComponent.vue'// 定义响应式引用,名字必须与模板中的 ref 值匹配
    const inputRef = ref(null)
    const childRef = ref(null)// 组件挂载后,引用才可用
    onMounted(() => {// 访问 DOM 元素if (inputRef.value) {inputRef.value.focus() // 让输入框自动获得焦点console.log('Input width:', inputRef.value.offsetWidth)}// 调用子组件的方法 (假设子组件暴露了 doSomething 方法)if (childRef.value) {childRef.value.doSomething()}
    })
    </script>

关键要点与注意事项

  1. 响应式引用 (ref) vs 模板引用 (ref attribute)

    • ref() 是一个函数,用于创建一个响应式引用对象。这个对象有一个 .value 属性,用来存储值(在这里是 DOM 元素或组件实例)。
    • 模板中的 ref="xxx" 是一个特殊的 attribute,它告诉 Vue 将这个元素/组件的引用注入到名字为 xxx 的响应式引用 (ref) 中。
    • 名字必须匹配const xxx = ref(null) 和 ref="xxx" 中的 xxx 必须完全相同。
  2. 初始值与访问时机

    • 通常将响应式引用初始化为 null (const myRef = ref(null)),因为在组件挂载前,DOM 元素或子组件实例还不存在。
    • 在 onMounted 生命周期钩子之前,引用的 .value 通常是 null。因为 DOM 渲染发生在 onMounted 之后。
    • 最佳实践:在 onMounted 或 onUpdated 钩子中访问引用,或者在事件处理函数中(确保元素已渲染)。
  3. 引用类型

    • DOM 元素:引用 .value 直接指向原生的 DOM 元素对象(如 HTMLInputElementHTMLDivElement 等),你可以调用其所有原生方法和属性。
    • 子组件:引用 .value 指向子组件的实例。你可以访问子组件的公开属性和方法(即在 setup 返回或在 <script setup> 中用 defineExpose 暴露的属性/方法)。
    <!-- ChildComponent.vue -->
    <script setup>
    import { ref } from 'vue'const count = ref(0)// 暴露给父组件的方法
    function increment() {count.value++
    }// 明确暴露哪些属性/方法给父组件
    defineExpose({increment,// count // 也可以暴露响应式数据,但需谨慎
    })
    </script>
  4. 访问子组件的 $el

    • 在 Vue 3 的 Composition API 中,子组件实例本身不直接是 DOM 元素。如果你需要访问子组件的根 DOM 元素,可以通过子组件实例的 $.vnode.el 属性(这是 Vue 内部的,不推荐直接依赖)或者让子组件通过 defineExpose 暴露其根元素的引用。
  5. v-for 中的模板引用

    • 当 ref 用在 v-for 内部的元素或组件上时,对应的引用将是一个包含相应数据的数组,顺序与 v-for 渲染的顺序一致。
    • 重要ref 不会随着 v-for 数据的更新而自动同步更新数组。如果数据列表变化(增删改),你需要手动管理这个引用数组,或者考虑使用其他模式(如 key + 计算属性)。
    <template><div v-for="(item, index) in list" :key="item.id" :ref="el => divs[index] = el">{{ item.text }}</div>
    </template><script setup>
    import { ref, reactive, onBeforeUpdate } from 'vue'const list = ref([{ id: 1, text: 'A' }, { id: 2, text: 'B' }])// 使用函数式 ref 回调来更灵活地收集引用
    const divs = ref([])// 在每次更新前重置引用数组,避免残留
    onBeforeUpdate(() => {divs.value = []
    })
    </script>
  6. 函数式 ref

    • 除了字符串 ref="xxx",你还可以传递一个函数 :ref="callback"。这个函数会在每次组件更新时被调用,接收 DOM 元素或组件实例作为参数。这在需要更精细控制引用收集逻辑时非常有用(如上面 v-for 的例子)。
    <template><input :ref="(el) => inputElement = el" />
    </template><script setup>
    import { ref } from 'vue'const inputElement = ref(null) // 函数内部会设置它// 或者更复杂的逻辑
    const setupInputRef = (el) => {if (el) {// 元素被挂载inputElement.value = el// 可以在这里做初始化操作} else {// 元素被卸载inputElement.value = null}
    }
    </script>
  7. TypeScript 支持

    • 在 TypeScript 中,你可以为模板引用提供精确的类型。
    <script setup lang="ts">
    import { ref, onMounted } from 'vue'
    import ChildComponent from './ChildComponent.vue'// 为 DOM 元素引用提供类型
    const inputRef = ref<HTMLInputElement | null>(null)// 为子组件引用提供类型 (需要导入组件类型)
    const childRef = ref<InstanceType<typeof ChildComponent> | null>(null)onMounted(() => {if (inputRef.value) {inputRef.value.focus() // TypeScript 知道这是 HTMLInputElement}if (childRef.value) {childRef.value.increment() // TypeScript 知道可以调用 increment}
    })
    </script>

Vue 3 的模板引用 (ref) 是一个强大且常用的特性,用于在父组件中直接操作 DOM 或子组件。核心是:

  1. 在模板中使用 ref="myName"
  2. 在 <script setup> 中使用 const myName = ref(null) 定义响应式引用。
  3. 在 onMounted 或之后访问 myName.value 来获取 DOM 元素或组件实例。
  4. 对于子组件,使用 defineExpose 来控制暴露的 API。
  5. 在 v-for 中使用时,引用是数组,需注意更新时机。
  6. 在 TypeScript 中提供精确类型。

原则:尽量通过响应式数据和 props/events 来进行组件通信,仅在确实需要直接 DOM 操作或调用特定方法时才使用模板引用。

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

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

相关文章

模式匹配自动机全面理论分析

模式匹配是什么 模式匹配是计算机科学中一个基础且重要的问题&#xff0c;广泛应用于文本编辑、信息检索、网络安全、生物信息学等多个领域。简单来说&#xff0c;模式匹配就是在一个主文本中查找一个或多个特定模式串的出现位置。随着计算机处理能力的提升和数据规模的扩大&am…

AI 搜索时代:引领变革,重塑您的 SEO 战略

随着谷歌转向人工智能驱动的答案&#xff0c;使用以关键字和反向链接为中心的过时和传统的 SEO 策略不再起到任何作用。 由于 Google AI Overviews 和零点击搜索的兴起&#xff0c;自然点击量正在下降&#xff0c;用户无需点击任何网站即可直接在 Google 的搜索结果页面上获得答…

【网站深入seo方法】

目录 ①对于更成熟的网站&#xff0c;简单的index.html的入口文件的seo已经无法满足&#xff0c;需要在商品详情不同商品被搜索时赋予不同的title和description。 ②通过设置站点所有页面都新增Canonical标签&#xff0c;指定规范链接地址给谷歌并规避联盟的重复内容页面。 ③…

ROS move_base 混合功能导航 RealSense D435i + 3D 点云地图 + 楼层切换 + 路径录制 + 路径规划

Mixed-Navigation 这个博客也是记录我们的一个开源项目&#xff0c;其作用是混合功能导航。由于现有的 Fast-Lio-Localization 只实现了定位功能&#xff0c;但对于路径规划和楼层切换没有具体实现&#xff0c;因此我们开出了这个仓库作为参考。该仓库的核心功能如下&#xff…

初识c语言————宏定义和调用

目录&#xff1a;一.不带参数的宏二.带参数宏一.不带参数的宏不带参数的宏是指用#define指令定义的简单文本替换规则&#xff0c;它没有参数列表&#xff0c;直接替换标识符为相应的文本其一般形式为&#xff1a;#define 宏名 文本例如&#xff1a;#define pi 3.14这个代…

数据结构:满二叉树 (Full Binary Tree) 和 完全二叉树 (Complete Binary Tree)

目录 重要的术语澄清 完美二叉树 (Perfect Binary Tree) 完全二叉树 (Complete Binary Tree) 大比拼 (Comparison) 相互关系的第一性推导 我们来深入探讨两种在算法中非常重要的、具有特定“形状”的二叉树&#xff1a;满二叉树 (Full Binary Tree) 和 完全二叉树 (Compl…

OpenJDK 17的C1和C2编译器实现中,方法返回前插入安全点(Safepoint Poll)的机制

OpenJDK 17 JIT编译器堆栈分析-CSDN博客 在OpenJDK 17的C1和C2编译器实现中&#xff0c;方法返回前插入安全点&#xff08;Safepoint Poll&#xff09;的机制主要涉及以下关键步骤&#xff0c;结合源代码进行分析&#xff1a; 1. 安全点轮询桩&#xff08;Safepoint Poll Stu…

【论文笔记】STORYWRITER: A Multi-Agent Framework for Long Story Generation

论文信息 论文标题&#xff1a;StoryWriter: A Multi-Agent Framework for Long Story Generation 论文作者&#xff1a;Haotian Xia, Hao Peng et al. (Tsinghua University) 论文链接&#xff1a;https://arxiv.org/abs/2506.16445 代码链接&#xff1a;https://github.com/…

Cohere 开发企业级大型语言模型(LLM)

Cohere 是一家专注于开发企业级大型语言模型&#xff08;LLM&#xff09;的公司。该公司推出了一系列名为 “Command” 的模型&#xff0c;其中最强大的 “Command A” 于今年三月首次亮相 Cohere 还提供嵌入模型&#xff0c;这是一种将文件转化为神经网络可以理解的紧凑数值形…

Rust Web框架Axum学习指南之入门初体验

一、准备阶段 确保已经安装 rust&#xff0c;开发环境使用 vscode 或者 rustrover 都可以。接着就可以创建项目&#xff0c;通过编辑器创建或者命令行创建都可以&#xff1a; cargo new axum-admin二、添加依赖 添加依赖如下&#xff1a; [package] name "axum-admin&quo…

autofit.js: 自动调整HTML元素大小的JavaScript库

&#x1f90d; 前端开发工程师、技术日更博主、已过CET6 &#x1f368; 阿珊和她的猫_CSDN博客专家、23年度博客之星前端领域TOP1 &#x1f560; 牛客高级专题作者、打造专栏《前端面试必备》 、《2024面试高频手撕题》、《前端求职突破计划》 &#x1f35a; 蓝桥云课签约作者、…

RocketMQ 命名服务器(NameServer)详解

&#x1f680; RocketMQ 命名服务器&#xff08;NameServer&#xff09;详解 NameServer 是 RocketMQ 架构中的轻量级路由发现服务&#xff0c;它不参与消息的收发&#xff0c;但承担着整个集群的“地址簿”和“导航系统”的关键角色。 理解 NameServer 的设计与工作原理&#…

代码随想录算法训练营四十三天|图论part01

深度优先搜索&#xff08;dfs&#xff09;理论基础 dfs就是可一个方向去搜直到尽头再换方向&#xff0c;换方向的过程就涉及到了回溯。 代码框架 因为dfs搜索可一个方向&#xff0c;并需要回溯&#xff0c;所以用递归的方式来实现是最方便的。 先来回顾一下回溯法的代码框架…

飞算JavaAI金融风控场景实践:从实时监测到智能决策的全链路安全防护

目录一、金融风控核心场景的技术突破1.1 实时交易风险监测系统1.1.1 高并发交易数据处理1.2 智能反欺诈系统架构1.2.1 多维度欺诈风险识别1.3 动态风控规则引擎1.3.1 风控规则动态管理二、金融风控系统效能升级实践2.1 风控模型迭代加速机制2.1.1 自动化特征工程结语&#xff1…

Vue 组件二次封装透传slots、refs、attrs、listeners

最近写了一个开源项目&#xff0c;有些地方需要二次封装&#xff0c;需要透传一些数据&#xff0c;需要注意的是ref&#xff0c;我这里使用俩种方式间接传递ref&#xff0c;具体如下&#xff1a; 使用&#xff1a; import VideoPlayer from ./index.jsVue.use(VideoPlayer)inde…

介绍大根堆小根堆

文章目录一、核心定义与结构特性示例&#xff08;以“数组存储堆”为例&#xff09;二、堆的两个核心操作1. 插入操作&#xff08;以小根堆为例&#xff09;2. 删除极值操作&#xff08;以小根堆为例&#xff0c;删除根节点的最小值&#xff09;三、小根堆 vs 大根堆&#xff1…

【Html网页模板】赛博朋克数据分析大屏网页

目录专栏导读✨ 项目概述&#x1f3a8; 设计理念&#x1f6e0;️ 技术架构核心技术栈设计模式&#x1f3af; 核心功能1. 视觉效果系统&#x1f308; 色彩体系2. 数据可视化模块&#x1f4ca; 主图表系统&#x1f4c8; 性能监控面板3. 实时数据流系统⚡ 数据流动画&#x1f4ca;…

【经典上穿突破】副图/选股指标,双均线交叉原理,对价格波动反应灵敏,适合捕捉短期启动点

【经典上穿突破】副图/选股指标&#xff0c;双均线交叉原理&#xff0c;对价格波动反应灵敏&#xff0c;适合捕捉短期启动点 这是一款结合短线与中线信号的趋势跟踪指标&#xff0c;通过双均线交叉原理捕捉股价突破时机&#xff0c;适用于个股分析和盘中选股。 核心功能模块&…

RK3568 NPU RKNN(四):RKNN-ToolKit2性能和内存评估

文章目录1、前言2、目标3、完整的测试程序4、运行测试程序5、程序拆解6、总结1、前言 本文仅记录本人学习过程&#xff0c;不具备教学指导意义。 2、目标 使用野火提供的示例程序&#xff0c;体验 RKNN-ToolKit2 在PC端使用连板推理&#xff0c;进行性能和内存评估。 3、完…

ASP.NET 上传文件安全检测方案

一、前端初步过滤&#xff08;防误操作&#xff09;<!-- HTML部分 --><input type"file" id"fileUpload" accept".jpg,.png,.pdf,.docx" /><button onclick"validateFile()">上传</button><script>func…