news 2026/4/18 12:13:25

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/4/15 11:25:45

用Sambert-HifiGan为播客节目自动生成多情感旁白

用Sambert-HifiGan为播客节目自动生成多情感旁白 引言&#xff1a;中文多情感语音合成的现实需求 在当前内容创作爆发的时代&#xff0c;播客、有声书、短视频配音等音频内容形式日益普及。然而&#xff0c;高质量的人声录制成本高、周期长&#xff0c;且难以实现情绪多样化表达…

作者头像 李华
网站建设 2026/4/18 10:08:19

无需GPU也能跑TTS:开源镜像CPU优化方案,响应速度提升300%

无需GPU也能跑TTS&#xff1a;开源镜像CPU优化方案&#xff0c;响应速度提升300% &#x1f3af; 背景与痛点&#xff1a;中文多情感语音合成的落地挑战 在智能客服、有声阅读、虚拟主播等场景中&#xff0c;高质量中文语音合成&#xff08;Text-to-Speech, TTS&#xff09; 已…

作者头像 李华
网站建设 2026/4/18 11:55:58

从零搭建语音合成平台:基于ModelScope镜像,支持并发100+请求

从零搭建语音合成平台&#xff1a;基于ModelScope镜像&#xff0c;支持并发100请求 &#x1f4cc; 背景与需求&#xff1a;为什么需要自建语音合成服务&#xff1f; 随着智能客服、有声阅读、虚拟主播等AI应用场景的爆发式增长&#xff0c;高质量的中文语音合成&#xff08;TTS…

作者头像 李华
网站建设 2026/4/18 12:04:49

客服中心智能化改造:自动播报+多情感切换提升满意度

客服中心智能化改造&#xff1a;自动播报多情感切换提升满意度 在现代客服系统中&#xff0c;语音交互的自然度与情感表达能力直接影响用户体验和满意度。传统的机械式语音播报已难以满足用户对“人性化服务”的期待。为此&#xff0c;基于中文多情感语音合成技术的智能化升级方…

作者头像 李华
网站建设 2026/4/18 8:53:19

从零开始部署图像转视频AI:开源镜像+GPU高效适配方案

从零开始部署图像转视频AI&#xff1a;开源镜像GPU高效适配方案 &#x1f4cc; 引言&#xff1a;为什么需要本地化部署图像转视频AI&#xff1f; 随着AIGC技术的爆发式发展&#xff0c;图像生成视频&#xff08;Image-to-Video, I2V&#xff09; 已成为内容创作、影视预演、广…

作者头像 李华
网站建设 2026/4/18 10:49:50

周末项目:用Llama Factory构建你的第一个AI诗人

周末项目&#xff1a;用Llama Factory构建你的第一个AI诗人 作为一个文学爱好者&#xff0c;你是否曾幻想过拥有一个能随时为你创作诗歌的AI助手&#xff1f;现在&#xff0c;借助Llama Factory这个强大的工具&#xff0c;即使没有任何深度学习背景&#xff0c;你也可以在几分钟…

作者头像 李华