news 2026/4/3 4:53:58

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/1 14:33:07

Llama Factory极限挑战:在边缘设备上实现实时微调

Llama Factory极限挑战&#xff1a;在边缘设备上实现实时微调 在工业物联网场景中&#xff0c;设备需要根据现场数据持续优化模型&#xff0c;但计算资源极其有限。本文将介绍如何利用Llama Factory这一开源低代码大模型微调框架&#xff0c;在边缘设备上实现实时微调。这类任务…

作者头像 李华
网站建设 2026/3/21 6:10:53

Markdown元数据驱动语音合成:结构化内容处理方案

Markdown元数据驱动语音合成&#xff1a;结构化内容处理方案 &#x1f4cc; 引言&#xff1a;从静态文本到情感化语音的演进 在智能语音交互日益普及的今天&#xff0c;高质量、多情感的中文语音合成&#xff08;TTS&#xff09; 已成为智能客服、有声阅读、虚拟主播等场景的…

作者头像 李华
网站建设 2026/4/2 15:49:37

零售收银自动化:CRNN OCR识别商品条码

零售收银自动化&#xff1a;CRNN OCR识别商品条码 引言&#xff1a;OCR技术在零售场景中的核心价值 在现代零售系统中&#xff0c;收银效率直接影响顾客体验与门店运营成本。传统人工扫描条码的方式不仅耗时&#xff0c;还容易因条码污损、光照不均或角度倾斜导致识别失败。随着…

作者头像 李华
网站建设 2026/3/16 18:02:54

揭秘Llama Factory:如何用预置镜像1小时完成大模型微调

揭秘Llama Factory&#xff1a;如何用预置镜像1小时完成大模型微调 作为一名独立开发者&#xff0c;你是否也遇到过这样的困扰&#xff1a;想为自己的小说创作一个AI助手&#xff0c;却被租用云服务器和配置环境的复杂流程劝退&#xff1f;今天我要分享的Llama Factory预置镜像…

作者头像 李华
网站建设 2026/3/30 14:51:41

Llama Factory企业级部署:高可用微调环境搭建指南

Llama Factory企业级部署&#xff1a;高可用微调环境搭建指南 对于科技公司的AI团队而言&#xff0c;为多个项目建立标准化的微调环境往往面临基础设施不统一的挑战。本文将介绍如何利用Llama Factory搭建高可用的大模型微调环境&#xff0c;实现云端灵活部署与一致性管理。这类…

作者头像 李华
网站建设 2026/3/23 18:59:12

wav格式兼容性最好?实测支持MP3/OGG等主流格式转换

wav格式兼容性最好&#xff1f;实测支持MP3/OGG等主流格式转换 &#x1f4d6; 项目简介 本镜像基于 ModelScope 经典的 Sambert-HifiGan&#xff08;中文多情感&#xff09; 模型构建&#xff0c;提供高质量、端到端的中文语音合成能力。模型具备细腻的情感表达能力&#xff0c…

作者头像 李华