关于Vue 3 <script setup> defineXXX API 总结

学!

目录

引言

在 Vue 3 的 <script setup> 中,有一系列以 define 开头的 API,用于简化组件的 props、事件、插槽、暴露接口等声明。它们是组合式 API 的语法糖,支持 TypeScript 类型推导,并显著减少模板和 JS 的冗余写法。

defineProps

  • 声明组件接收的 props。
  • 支持 TypeScript 类型推导。
  • 取代了传统的 props: {} 配置。
typescript 复制代码
<script setup lang="ts">
interface Props {
  title: string
  count?: number
}

const props = defineProps<Props>()
</script>

<template>
  <h1>{{ props.title }} - {{ props.count }}</h1>
</template>

可以直接解构:

typescript 复制代码
const { title, count } = defineProps<{ title: string, count?: number }>()

父组件传入的值,会自动映射到 props 上。

defineEmits

  • 声明组件可触发的事件。
  • 在 TypeScript 下自动检查事件名称和参数类型。
  • 取代了传统的 this.$emit,并提供类型安全。
typescript 复制代码
<script setup lang="ts">
const emit = defineEmits<{
  (e: 'update', value: number): void
  (e: 'delete'): void
}>()

function handleClick() {
  emit('update', 42)
  emit('delete')
}
</script>

emit 对象用于触发事件。

提供类型检查,避免事件名写错或参数类型错误。

defineExpose

  • 显式暴露子组件的属性或方法给父组件。
  • 父组件通过 ref 获取子组件实例时,只有 defineExpose 暴露的内容可访问。
  • 用于 <script setup>,是 setup(props, { expose }) 的语法糖。
typescript 复制代码
<script setup lang="ts">
import { ref } from 'vue'

const count = ref(0)
function increment() { count.value++ }

defineExpose({ count, increment })
</script>

父组件访问方式:

typescript 复制代码
const childRef = ref(null)
childRef.value.increment()
console.log(childRef.value.count)

defineSlots(Vue 3.4+)

  • 声明组件可接收的插槽类型。
  • 支持 TypeScript 类型推导。
  • 用于 <script setup> 下的插槽类型安全。
typescript 复制代码
<script setup lang="ts">
const slots = defineSlots<{
  header?: () => any
  footer?: () => any
}>()
</script>

<template>
  <header v-if="slots.header">{{ slots.header() }}</header>
  <footer v-if="slots.footer">{{ slots.footer() }}</footer>
</template>

defineAsyncComponent

  • 定义异步加载组件。
  • 按需加载,减少初始包体积。
  • 可配置加载组件、失败组件、超时等。
typescript 复制代码
import { defineAsyncComponent } from 'vue'

const AsyncComp = defineAsyncComponent(() => import('./MyComp.vue'))

可与 <component :is="AsyncComp" /> 配合动态渲染。

defineModel(Vue 3.3+)

  • 简化 v-model 双向绑定。
  • 支持 TypeScript 类型推导。
  • 自动生成 modelValue 和 update:modelValue 的类型绑定。
typescript 复制代码
<script setup lang="ts">
defineModel('modelValue', Number)
</script>

<template>
  <input v-model="modelValue" />
</template>

数据流可视化

typescript 复制代码
          ┌───────────── Parent ─────────────┐
          │                                  │
          │ props (父传子)                    │
          │ ┌───────────────┐                │
          │ │ defineProps   │<───父组件传值───┐
          │ └───────────────┘              	 │
          │                                	 │
          │ emits (子传父)               	 │
          │ ┌───────────────┐                │
          │ │ defineEmits   │ ───触发事件────>│
          │ └───────────────┘                │
          │                                  │
          │ ref 获取子组件实例                 │
          │ ┌───────────────┐                │
          │ │ defineExpose  │───暴露方法/属性─>│
          │ └───────────────┘                │
          │                                  │
          │ slots (父传子内容)                 │
          │ ┌───────────────┐                │
          │ │ defineSlots   │<─ 父组件传slot──┐
          │ └───────────────┘               │
          └─────────────────────────────────┘
  • 父 → 子:props(defineProps)、slots(defineSlots)
  • 子 → 父:emits(defineEmits)
  • 父通过 ref → 子:defineExpose 暴露的内容
  • 双向绑定:defineModel
相关推荐
恋猫de小郭1 天前
Amper 正式转正 Kotlin Toolchain ,Gradle 未来何去何从
android·前端·flutter
敲代码的彭于晏1 天前
Bean 生命周期完全图解:前端同学也能看懂的 Spring 核心机制
java·前端·后端
IT_陈寒1 天前
Redis内存飙升的锅,原来是我没搞懂这个过期策略
前端·人工智能·后端
云浪1 天前
前端二进制数组完全指南:ArrayBuffer、TypedArray、DataView 一次讲透
前端·javascript
张风捷特烈1 天前
Flutter 类库大揭秘#02 | path_provider 各平台实现
前端·flutter
铁皮饭盒1 天前
26年bunjs, elysia+pg一把梭, redis都省了
前端·javascript·后端
lichenyang4532 天前
Docker 学习笔记(一):为什么需要镜像、容器和仓库?
前端
kyriewen2 天前
别再对着 TypeScript 报错发呆了:我把 10 个最常见的红色波浪线翻译成了人话
前端·javascript·typescript
IT_陈寒2 天前
SpringBoot自动配置的坑,我的API突然就404了
前端·人工智能·后端