原文 👉 Props | Vue.js

此章节假设你已经看过了组件基础。若你还不了解组件是什么,请先阅读该章节(组件基础)。

Props 声明

一个组件需要显式声明它所接受的 props,这样 Vue 才能知道外部传入的哪些是 props,哪些是透传 attribute (关于透传 attribute,我们会在专门的章节中讨论)。

defineProps() 宏

  1. <script setup>
  2. import B from "./B.vue"
  3. </script>
  4. <template>
  5. <B foo="123" />
  6. </template>
  1. <script setup>
  2. const props = defineProps(['foo'])
  3. console.log(props.foo) // => 123
  4. </script>
  5. <template>
  6. <h1>B.vue {{ props.foo }}</h1>
  7. </template>

image.png

在使用 <script setup> 的单文件组件中,props 可以使用 defineProps() 宏来声明。

通过 defineProps() 宏来声明的 props 会自动注入到组件实例中,我们可以直接在 template 中访问这些 props

props 选项

A.vue 的内容保持不变:

  1. <script setup>
  2. import B from "./B.vue"
  3. </script>
  4. <template>
  5. <B foo="123" />
  6. </template>
  1. <script>
  2. export default {
  3. props: ['foo']
  4. }
  5. </script>
  6. <template>
  7. <h1>B.vue {{ foo }}</h1>
  8. </template>

image.png

在没有使用 <script setup> 的组件中,prop 可以使用 props 选项来声明。

传递给 defineProps() 的参数和提供给 props 选项的值是相同的,两种声明方式背后其实使用的都是 prop 选项

通过这种非 setup 的写法添加的属性,也会自动注入到组件实例身上,因此,最终我们看到的页面渲染结果并没有发生变化,在模板 template 中我们可以直接通过属性名 foo 获取到父组件 A.vue 传递过来的属性值 123。

若想要在 setup 函数中获取到 props,那么我们可以通过 setup 函数的第一个参数来接收:

  1. <script>
  2. export default {
  3. props: ['foo'],
  4. setup(props) { // setup() 接收 props 作为第一个参数
  5. console.log(props.foo) // => 123
  6. return {
  7. f: props.foo
  8. }
  9. }
  10. }
  11. </script>
  12. <template>
  13. <h1>B.vue</h1>
  14. <h2>f: {{ f }}</h2>
  15. <h2>foo: {{ foo }}</h2>
  16. </template>

image.png

对象

  1. <script setup>
  2. import B from "./B.vue"
  3. </script>
  4. <template>
  5. <B title="test" :likes="123" />
  6. </template>
  1. <script setup>
  2. const {
  3. title,
  4. likes
  5. } = defineProps({
  6. title: String,
  7. likes: Number
  8. })
  9. console.log(typeof title) // => string
  10. console.log(typeof likes) // => number
  11. </script>
  12. <template>
  13. <h1>B.vue</h1>
  14. <h2>title: {{ title }}</h2>
  15. <h2>likes: {{ likes }}</h2>
  16. </template>
  1. <script>
  2. export default {
  3. props: {
  4. title: String,
  5. likes: Number
  6. }
  7. }
  8. </script>
  9. <template>
  10. <h1>B.vue</h1>
  11. <h2>title: {{ title }}</h2>
  12. <h2>likes: {{ likes }}</h2>
  13. </template>

image.png

除了使用字符串数组来声明 prop 外,还可以使用对象的形式

对于以对象形式声明中的每个属性:

  • key 是 prop 的名称
  • value 是该 prop 预期类型的构造函数

比如,如果要求一个 prop 的值是 number 类型,则可使用 Number 构造函数作为其声明的值。

对象形式的 props 声明不仅可以一定程度上作为组件的文档,而且如果其他开发者在使用你的组件时传递了错误的类型,也会在浏览器控制台中抛出警告。我们将在本章节稍后进一步讨论有关 prop 校验的更多细节。

传递 prop 的细节

Prop 名字格式

如果一个 prop 的名字很长,应使用 camelCase 形式,因为它们是合法的 JavaScript 标识符,可以直接在模板的表达式中使用,也可以避免在作为属性 key 名时必须加上引号。

  1. <script setup>
  2. defineProps({
  3. greetingMessage: String
  4. })
  5. </script>
  6. <template>
  7. <h1>B.vue {{ greetingMessage }}</h1>
  8. </template>

不推荐

  1. <script setup>
  2. import B from "./B.vue"
  3. </script>
  4. <template>
  5. <B greetingMessage="hello" />
  6. </template>

译:greeting 问候语

虽然理论上你也可以在向子组件传递 props 时使用 camelCase 形式,但实际上为了和 HTML attribute 对齐,我们通常会将其写为 kebab-case 形式

推荐

  1. <script setup>
  2. import B from "./B.vue"
  3. </script>
  4. <template>
  5. <B greeting-message="hello" />
  6. </template>

image.png

对于组件名我们推荐使用 PascalCase,因为这提高了模板的可读性,能帮助我们区分 Vue 组件和原生 HTML 元素。然而对于传递 props 来说,使用 camelCase 并没有太多优势,因此我们推荐更贴近 HTML 的书写风格

静态 vs. 动态 Prop

静态写死
静态:<BlogPost title="My journey with Vue" />
这种写法 title="xxx" 是静态绑定 prop,相当于直接将写死的 xxx 字符串作为 title 的属性传递过去

动态变量
动态:<BlogPost :title="post.title" />
这种写法 :title="xxx" 或者 v-bind:title="xxx" 是动态绑定 prop,这种写法传递的内容是 xxx 变量或具体的值(不一定是字符串),它可以是动态的变量,也可以是写死的字面量
与静态绑定 prop 不同的是:这种写法绑定的值 xxx 不一定是字符串;若采用静态的写法,那么 xxx 一定是字符串。

  1. <script setup>
  2. import B from "./B.vue"
  3. const post = {
  4. likes: 123,
  5. isPublished: true,
  6. commentIds: [11, 22, 33],
  7. author: {
  8. name: "xxx",
  9. company: "hahaha",
  10. },
  11. }
  12. </script>
  13. <template>
  14. <div style="display: flex">
  15. <!-- 不传递任何值,默认 default -->
  16. <B />
  17. <!-- 都使用字面量传递 -->
  18. <B
  19. title="字面量"
  20. :likes="42"
  21. :is-published="false"
  22. :comment-ids="[234, 266, 273]"
  23. :author="{
  24. name: 'Veronica',
  25. company: 'Veridian Dynamics',
  26. }"
  27. />
  28. <!-- 多使用变量传递 -->
  29. <B
  30. title="变量"
  31. :likes="post.likes"
  32. :is-published="post.isPublished"
  33. :comment-ids="post.commentIds"
  34. :author="post.author"
  35. />
  36. <!-- 直接使用 v-bind 来传递 -->
  37. <B title="v-bind" v-bind="post" />
  38. </div>
  39. </template>
  1. <script setup>
  2. const props = defineProps({
  3. title: {
  4. type: String,
  5. default: "default",
  6. },
  7. likes: {
  8. type: Number,
  9. default: 0,
  10. },
  11. isPublished: {
  12. type: Boolean,
  13. default: false,
  14. },
  15. commentIds: {
  16. type: Array,
  17. validator: (value) =>
  18. Array.isArray(value) && value.every((v) => typeof v === "number"),
  19. default: () => [],
  20. },
  21. author: {
  22. type: Object,
  23. validator: (value) =>
  24. value &&
  25. typeof value === "object" &&
  26. "name" in value &&
  27. "company" in value,
  28. default: () => ({ name: "", company: "" }),
  29. },
  30. })
  31. </script>
  32. <template>
  33. <div>
  34. <h1>{{ title }}</h1>
  35. <pre>{{ props }}</pre>
  36. </div>
  37. </template>
  38. <style scoped>
  39. pre {
  40. width: 300px;
  41. text-align: left;
  42. border: 1px solid #ddd;
  43. padding: 1rem;
  44. border-radius: 1rem;
  45. color: #fff;
  46. background: #292d3e;
  47. }
  48. </style>

image.png

  1. <!-- 写法1:多使用变量传递 -->
  2. <B
  3. title="变量"
  4. :likes="post.likes"
  5. :is-published="post.isPublished"
  6. :comment-ids="post.commentIds"
  7. :author="post.author"
  8. />
  9. <!-- 写法2:直接使用 v-bind 来传递 -->
  10. <B title="v-bind" v-bind="post" />

写法 1 和 写法 2 是完全等效的,其中写法 2 是:使用一个对象绑定多个 prop,如果你想要将一个对象的所有属性都当作 props 传入,你可以使用没有参数的 v-bind

单向数据流

所有的 props 都遵循着单向绑定原则,props 因父组件的更新而变化,自然地将新的状态向下流往子组件,而不会逆向传递。这避免了子组件意外修改父组件的状态的情况,不然应用的数据流将很容易变得混乱而难以理解。

另外,每次父组件更新后,所有的子组件中的 props 都会被更新到最新值,这意味着你不应该在子组件中去更改一个 prop。若你这么做了,Vue 会在控制台上向你抛出警告:

  1. <script setup>
  2. import B from "./B.vue"
  3. </script>
  4. <template>
  5. <B foo="123" />
  6. </template>
  1. <script setup>
  2. const props = defineProps(['foo'])
  3. // ❌ 警告!prop 是只读的!
  4. props.foo = 'bar'
  5. </script>
  6. <template>
  7. <h1>B.vue {{ props.foo }}</h1>
  8. </template>

image.png

记住一句话:谁的数据谁负责
这里的“谁”代指“组件”,“负责”代指“写”,简言之就是,组件 A 的数据只能由组件 A 修改,如果组件 A 将数据作为属性传递给了其它组件,比如组件 B、C,那么组件 B、C 只由读的份,没有写的份。

修改 prop

导致你想要更改一个 prop 的需求通常来源于以下两种场景:

  1. prop 被用于传入初始值
  2. 需要对传入的 prop 值做进一步的转换

(1)prop 被用于传入初始值

prop 被用于传入初始值;而子组件想在之后将其作为一个局部数据属性。在这种情况下,最好是新定义一个局部数据属性,从 props 上获取初始值即可:

  1. <script setup>
  2. import B from "./B.vue"
  3. </script>
  4. <template>
  5. <B :initial-counter="0" />
  6. </template>
  1. <script setup>
  2. import { ref } from "vue"
  3. const props = defineProps(['initialCounter'])
  4. // 计数器只是将 props.initialCounter 作为初始值
  5. // 像下面这样做就使 prop 和后续更新无关了
  6. const counter = ref(props.initialCounter)
  7. </script>
  8. <template>
  9. <h1>B.vue {{ counter }}</h1>
  10. <button @click="counter++">increase</button>
  11. </template>

image.png

const counter = ref(props.initialCounter)
将父组件传递过来的 initialCounter 属性的值取出来,丢到 counter 变量中,然后维护 counter 变量,这是非常常见的做法,并且这么做是完全没问题的。

  • initialCounter 是 A 的数据
  • counter 是 B 的数据

我们并没有直接修改 initialCounter,而是修改 counter,符合单向数据流规范。

(2)需要对传入的 prop 值做进一步的转换

需要对传入的 prop 值做进一步的转换。在这种情况中,最好是基于该 prop 值定义一个计算属性

  1. <script setup>
  2. import B from "./B.vue"
  3. </script>
  4. <template>
  5. <B size=" Size " />
  6. </template>
  1. <script setup>
  2. import { computed } from "vue"
  3. const props = defineProps(['size'])
  4. // 该 prop 变更时计算属性也会自动更新
  5. const normalizedSize = computed(() => props.size.trim().toLowerCase())
  6. </script>
  7. <template>
  8. <h1>B.vue {{ normalizedSize }} {{ normalizedSize.length }} {{ size.length }}</h1>
  9. </template>

image.png

引用类型 prop

当对象或数组作为 props 被传入时,虽然子组件无法更改 props 绑定,但仍然可以更改对象或数组内部的值。这是因为 JavaScript 的对象和数组是按引用传递,而对 Vue 来说,禁止这样的改动,虽然可能生效,但有很大的性能损耗,比较得不偿失。

这种更改的主要缺陷是它允许了子组件以某种不明显的方式影响父组件的状态,可能会使数据流在将来变得更难以理解。在最佳实践中,你应该尽可能避免这样的更改,除非父子组件在设计上本来就需要紧密耦合。在大多数场景下,子组件应该抛出一个事件来通知父组件做出改变。

反模式

  1. <script setup>
  2. import B from "./B.vue"
  3. import C from "./C.vue"
  4. import { ref } from "vue"
  5. const dataA = ref({
  6. a: 1,
  7. b: 2,
  8. })
  9. </script>
  10. <template>
  11. <B :data="dataA" />
  12. <C :data="dataA" />
  13. </template>
  1. <script setup>
  2. defineProps(['data'])
  3. </script>
  4. <template>
  5. <h1>B.vue {{ data }}</h1>
  6. </template>
  1. <script setup>
  2. defineProps(["data"])
  3. </script>
  4. <template>
  5. <h1>C.vue {{ data }}</h1>
  6. <button
  7. @click="
  8. () => {
  9. data.a = 11
  10. console.log('C clicked', data)
  11. }
  12. "
  13. >
  14. data.a = 11
  15. </button>
  16. </template>

image.png

上述提供的 demo 是一个反模式,不符合单向数据流,因为我们在 C 组件中,直接修改了来自父组件 A 传递过来的数据 data。

最佳实践
正确的做法应该是:

  1. 监听按钮点击事件
  2. 按钮被点击后通知父组件 A(因为数据来源自 A)
  3. 由 A 来决定是否修改 data
  1. <script setup>
  2. defineProps(["data"])
  3. defineEmits(['updateData'])
  4. </script>
  5. <template>
  6. <h1>C.vue {{ data }}</h1>
  7. <button
  8. @click="$emit('updateData', 'a', 11)"
  9. >
  10. data.a = 11
  11. </button>
  12. </template>
  1. <script setup>
  2. import B from "./B.vue"
  3. import C from "./C.vue"
  4. import { ref } from "vue"
  5. const dataA = ref({
  6. a: 1,
  7. b: 2,
  8. })
  9. const handleUpdate = (k, v) => {
  10. console.log(k, v)
  11. // 由 A 决定是否修改 data
  12. dataA.value[k] = v
  13. }
  14. </script>
  15. <template>
  16. <B :data="dataA" />
  17. <C :data="dataA" @update-data="handleUpdate" />
  18. </template>

image.png

从最终结果来看都是一样的,但是后者这种做法才是符合单向数据流规范的。试想一下其实不难理解,dataA 数据存在于 A 组件中,该数据是如何被使用的,只有 A 组件最为清楚,我们如果在某个子组件中直接把它给修改了,如果这个数据正在被其它组件使用,那么很可能会影响到其它组件造成影响。这种行为所导致的问题,调试起来成本极高,是不符合规范的。

Prop 校验

校验选项

Vue 组件可以更细致地声明对传入的 props 的校验要求。比如我们上面已经看到过的类型声明,如果传入的值不满足类型要求,Vue 会在浏览器控制台中抛出警告来提醒使用者。这在开发给其他开发者使用的组件时非常有用。

要声明对 props 的校验,你可以向 defineProps() 宏提供一个带有 props 校验选项的对象,例如:

  1. <script setup>
  2. const props = defineProps({
  3. // 基础类型检查
  4. // (给出 `null` 和 `undefined` 值则会跳过任何类型检查)
  5. propA: Number,
  6. // 多种可能的类型
  7. propB: [String, Number],
  8. // 必传,且为 String 类型
  9. propC: {
  10. type: String,
  11. required: true
  12. },
  13. // Number 类型的默认值
  14. propD: {
  15. type: Number,
  16. default: 100
  17. },
  18. // 对象类型的默认值
  19. propE: {
  20. type: Object,
  21. // 对象或数组的默认值
  22. // 必须从一个工厂函数返回。
  23. // 该函数接收组件所接收到的原始 prop 作为参数。
  24. default(rawProps) {
  25. return { message: 'hello' }
  26. }
  27. },
  28. // 自定义类型校验函数
  29. propF: {
  30. validator(value) {
  31. // The value must match one of these strings
  32. return ['success', 'warning', 'danger'].includes(value)
  33. }
  34. },
  35. // 函数类型的默认值
  36. propG: {
  37. type: Function,
  38. // 不像对象或数组的默认,这不是一个
  39. // 工厂函数。这会是一个用来作为默认值的函数
  40. default() {
  41. return 'Default function'
  42. }
  43. }
  44. })
  45. </script>
  46. <template>
  47. <h1>B.vue</h1>
  48. <pre>{{ props }}</pre>
  49. </template>
  50. <style scoped>
  51. pre {
  52. width: 300px;
  53. text-align: left;
  54. border: 1px solid #ddd;
  55. padding: 1rem;
  56. border-radius: 1rem;
  57. color: #fff;
  58. background: #292d3e;
  59. }
  60. </style>
  1. <script setup>
  2. import B from "./B.vue"
  3. </script>
  4. <template>
  5. <B :propA='1' propC='123' propF='success' />
  6. </template>

image.png

当 prop 的校验失败后,Vue 在开发模式下会抛出一个控制台警告

<B />
如果缺少必填 required 属性,那么控制台会显示如下 Missing required prop 警告信息:
image.png

<B propC='123' propF='' />
如果填写的属性无法通过校验,那么控制台会提示我们 Invalid prop 无效属性:
image.png

<B propA='1' propC='123' propF='success' />
如果我们传递的数据类型错误,会提示类似 Expected Number with value 1, got String with value “1”.(期待的类型是 Number 类型 1,但是获取到的是一个 String 类型 “1”) 这样的警告信息
image.png

:::success TIP
**defineProps()** 宏中的参数不可以访问 **<script setup>** 中定义的其他变量,因为在编译时整个表达式都会被移到外部的函数中。 :::

细节:

  • 所有 prop 默认都是可选的,除非声明了 required: true
  • Boolean 外的未传递的可选 prop 将会有一个默认值 undefined
  • Boolean 类型的未传递 prop 将被转换为 false
    • 这可以通过为它设置 default 来更改
    • 例如:设置为 default: undefined 将与非布尔类型的 prop 的行为保持一致
  • 如果声明了 default 值,那么在 prop 的值被解析为 undefined 时,无论 prop 是未被传递还是显式指明的 undefined,都会改为 default

如果使用了基于类型的 prop 声明 ,Vue 会尽最大努力在运行时按照 prop 的类型标注进行编译。举例来说,defineProps<{ msg: string }> 会被编译为 { msg: { type: String, required: true }}

校验选项中的 type 可以是下列这些原生构造函数:

  • String
  • Number
  • Boolean
  • Array
  • Object
  • Date
  • Function
  • Symbol

instanceof

另外,type 也可以是自定义的类或构造函数,Vue 将会通过 instanceof 来检查类型是否匹配。

  1. export class Person {
  2. constructor(firstName, lastName) {
  3. this.firstName = firstName
  4. this.lastName = lastName
  5. }
  6. }
  1. <script setup>
  2. import B from "./B.vue"
  3. </script>
  4. <template>
  5. <B author="123" />
  6. </template>
  1. <script setup>
  2. import { Person } from "./Person.js"
  3. defineProps({
  4. author: Person
  5. // Vue 会通过 instanceof Person 来校验 author prop 的值是否是 Person 类的一个实例。
  6. })
  7. </script>
  8. <template>
  9. <h1>B.vue {{ author }}</h1>
  10. </template>

image.png

  1. <script setup>
  2. import B from "./B.vue"
  3. </script>
  4. <template>
  5. <B :author="{
  6. firstName: 'da',
  7. lastName: 'huyou'
  8. }" />
  9. </template>

image.png

  1. <script setup>
  2. import { Person } from "./Person.js"
  3. import B from "./B.vue"
  4. </script>
  5. <template>
  6. <B :author="new Person('da', 'huyou')" />
  7. </template>

image.png

Boolean 类型转换

为了更贴近原生 boolean attributes 的行为,声明为 Boolean 类型的 props 有特别的类型转换规则。以带有如下声明的 组件为例:

  1. <script setup>
  2. defineProps({
  3. disabled: Boolean
  4. })
  5. </script>
  6. <template>
  7. <h1>B.vue {{ disabled }}</h1>
  8. </template>
  1. <script setup>
  2. import B from "./B.vue"
  3. </script>
  4. <template>
  5. <!-- 等同于传入 :disabled="true" -->
  6. <B disabled />
  7. <!-- 等同于传入 :disabled="false" -->
  8. <B />
  9. </template>

image.png

  1. <script setup>
  2. defineProps({
  3. disabled: [Boolean, Number]
  4. })
  5. </script>
  6. <template>
  7. <h1>B.vue {{ disabled }}</h1>
  8. </template>

页面的最终渲染结果保持不变,无论声明类型的顺序如何,Boolean 类型的特殊转换规则都会被应用。