news 2026/1/11 8:07:59

Vue 3中:setup 函数接收两个重要参数:props 和 context

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
Vue 3中:setup 函数接收两个重要参数:props 和 context

props是响应式的只读对象,需通过defineProps声明,可使用toRefs保持解构后的响应性。


context包含三个非响应式属性:attrs(未声明的属性)、slots(插槽内容)和emit(触发事件)。


在<script setup>中推荐使用useAttrs()、useSlots()和defineEmits()。


注意事项包括setup执行时机、响应式丢失风险及TypeScript支持。


最佳实践建议优先使用<script setup>语法糖,配合类型声明和emit事件通信,充分利用Vue3响应式系统。


在 Vue3 的 Composition API 中,setup函数接收两个重要参数:propscontext


1.props 参数

props是响应式的对象,包含组件接收的所有 prop。


特性:

  • 响应式:当父组件传递的 prop 变化时,props会自动更新

  • 只读:不能在子组件中直接修改(会触发警告)

  • 需要显式声明:必须通过definePropsprops选项声明


基本使用:

vue

<script setup> import { toRefs, watch } from 'vue' // 声明 props const props = defineProps({ title: String, count: { type: Number, default: 0 } }) // 直接使用 console.log(props.title) // 监听 props 变化 watch(() => props.count, (newVal) => { console.log('count changed:', newVal) }) // 解构 props(会失去响应性) const { title } = props // ❌ 非响应式 // 保持响应性的解构 const { title, count } = toRefs(props) // ✅ 响应式 </script>

2.context 参数

context是一个普通对象(非响应式),包含三个组件属性。

context 包含三个属性:


a)attrs

vue

<script setup> import { useAttrs } from 'vue' // 方式一:通过 setup 参数 setup(props, context) { console.log(context.attrs) // 所有未在 props 中声明的 attribute } // 方式二:在 <script setup> 中使用 useAttrs() const attrs = useAttrs() console.log(attrs.class) // 获取 class 属性 console.log(attrs.onClick) // 获取事件监听器 </script>

  • 包含所有未在 props 中声明的 attribute

  • 包括classstyle、事件监听器等

  • 非响应式,但会自动更新


b)slots

vue

<script setup> import { useSlots } from 'vue' // 方式一:通过 setup 参数 setup(props, context) { // 检查插槽是否存在 if (context.slots.default) { // 渲染插槽内容 return () => context.slots.default() } } // 方式二:在 <script setup> 中使用 useSlots() const slots = useSlots() console.log(slots.default) // 默认插槽 console.log(slots.header) // 具名插槽 </script>

  • 包含所有插槽内容的函数

  • 用于渲染插槽内容或检查插槽是否存在


c)emit

vue

<script setup> import { defineEmits } from 'vue' // 方式一:通过 setup 参数 setup(props, context) { const handleClick = () => { context.emit('update', newValue) } } // 方式二:在 <script setup> 中使用 defineEmits() const emit = defineEmits(['update', 'delete']) const handleClick = () => { emit('update', { id: 1, value: 'new' }) emit('delete', 1) } </script>

  • 用于触发自定义事件

  • 推荐使用defineEmits进行声明


3.完整示例

选项式 API 写法:

vue

<script> import { toRefs, watch } from 'vue' export default { props: ['title', 'count'], emits: ['update-count'], setup(props, context) { // 使用 props const { title, count } = toRefs(props) // 使用 attrs console.log('所有 attributes:', context.attrs) // 使用 slots const hasHeaderSlot = !!context.slots.header // 使用 emit const increment = () => { context.emit('update-count', count.value + 1) } // 监听 props 变化 watch(count, (newVal) => { console.log('Count updated:', newVal) }) return { title, count, increment, hasHeaderSlot } } } </script>

<script setup>语法糖写法:

vue

<script setup> import { toRefs, watch, useAttrs, useSlots } from 'vue' // 声明 props const props = defineProps({ title: String, count: Number }) // 声明 emits const emit = defineEmits(['update-count']) // 使用 attrs 和 slots const attrs = useAttrs() const slots = useSlots() // 响应式解构 props const { title, count } = toRefs(props) // 检查插槽 const hasHeaderSlot = !!slots.header // 方法 const increment = () => { emit('update-count', count.value + 1) } // 监听器 watch(count, (newVal) => { console.log('Count updated:', newVal) }) </script> <template> <div :class="attrs.class"> <slot name="header" v-if="hasHeaderSlot" /> <h2>{{ title }}</h2> <p>Count: {{ count }}</p> <button @click="increment">Increment</button> <slot /> </div> </template>

4.注意事项

  1. setup 执行时机:在beforeCreate之前执行,此时无法访问this

  2. 响应式丢失:直接解构 props 会丢失响应性,使用toRefstoRef

  3. attrs 非响应式:但会自动更新,无需担心

  4. TypeScript 支持


vue

<script setup lang="ts"> interface Props { title: string count?: number } const props = defineProps<Props>() const emit = defineEmits<{ (e: 'update', value: number): void }>() </script>

5.最佳实践

  1. 优先使用<script setup>语法糖

  2. 使用definePropsdefineEmits进行声明

  3. 需要响应式解构时使用toRefs

  4. 避免直接修改 props,使用 emit 触发事件

  5. 使用 TypeScript 增强类型安全


通过合理使用这两个参数,可以更好地组织组件逻辑,并充分利用 Vue3 的响应式系统。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/1/9 15:03:45

Sambert-HifiGan+LangChain:快速构建企业级语音交互系统

Sambert-HifiGanLangChain&#xff1a;快速构建企业级语音交互系统 &#x1f4cc; 引言&#xff1a;为什么需要高质量的中文多情感语音合成&#xff1f; 在智能客服、虚拟主播、有声阅读和无障碍服务等场景中&#xff0c;自然、富有情感的中文语音合成&#xff08;TTS&#xff…

作者头像 李华
网站建设 2026/1/9 14:58:11

Llama Factory微调对比:8种不同配置的性能实测

Llama Factory微调对比&#xff1a;8种不同配置的性能实测 作为一名技术博主&#xff0c;我最近在准备一篇关于不同微调策略效果的深度文章时&#xff0c;遇到了一个棘手的问题&#xff1a;如何确保所有实验都在统一的环境中进行公平比较&#xff1f;经过多次尝试&#xff0c;…

作者头像 李华
网站建设 2026/1/9 14:57:51

Sambert-HifiGan性能基准测试:不同硬件配置对比

Sambert-HifiGan性能基准测试&#xff1a;不同硬件配置对比 &#x1f4ca; 引言&#xff1a;为何需要多硬件性能评测&#xff1f; 随着语音合成技术在智能客服、有声阅读、虚拟主播等场景的广泛应用&#xff0c;端到端中文多情感语音合成模型的部署效率与推理性能成为工程落地的…

作者头像 李华
网站建设 2026/1/9 14:56:30

Unity ECS 工作流程详解:从子场景到高性能系统

作者:硬汉小李 发布时间:2026年1月9日 适用 Unity 版本:Unity 2023 LTS 及以上(DOTS / Entities Package) 目录 前言 一、创建子场景(SubScene) 二、定义 ECS 组件 三、创建实体(Entities) 1. 编辑期创建(通过子场景烘焙) 2. 运行时创建 四、实现系统(System…

作者头像 李华
网站建设 2026/1/9 14:56:26

CRNN+OpenCV双剑合璧:打造更智能的文字识别系统

CRNNOpenCV双剑合璧&#xff1a;打造更智能的文字识别系统 &#x1f4d6; 项目简介 在数字化转型加速的今天&#xff0c;OCR&#xff08;Optical Character Recognition&#xff0c;光学字符识别&#xff09; 技术已成为信息自动化提取的核心工具。无论是发票、合同、身份证件&…

作者头像 李华
网站建设 2026/1/10 17:05:31

Sambert-HifiGan在公共广播系统中的语音合成应用

Sambert-HifiGan在公共广播系统中的语音合成应用 引言&#xff1a;中文多情感语音合成的现实需求 随着智能语音技术的快速发展&#xff0c;高质量、自然流畅的中文语音合成&#xff08;TTS&#xff09; 已成为公共服务领域的重要基础设施。尤其在公共广播系统中——如地铁报站、…

作者头像 李华