组合式 API:setup() {#composition-api-setup}

:::info 注意 这篇文档所讲的是组件 setup 选项的使用方式。如果你正在搭配单文件组件使用组合式 API,建议使用更简洁好用的 <script setup> 语法糖。 :::

setup() 这个钩子在以下情况下,作为组件中使用组合式 API 的入口。

  1. 不搭配构建步骤使用组合 API
  2. 在选项式 API 组件中集成基于组合式 API 的代码。

基本使用 {#basic-usage}

我们可以使用 响应性 API 来声明响应式的状态,并可以将其在 setup() 函数的返回值对象上,以此暴露给模板。在其他的选项中,返回值对象上的属性同样会作为组件实例的属性:

  1. <script>
  2. import { ref } from 'vue'
  3. export default {
  4. setup() {
  5. const count = ref(0)
  6. // 暴露给模板和 API 钩子的其他选项
  7. return {
  8. count
  9. }
  10. },
  11. mounted() {
  12. console.log(this.count) // 0
  13. }
  14. }
  15. </script>
  16. <template>
  17. <button @click="count++">{{ count }}</button>
  18. </template>

注意从 setup 返回的 ref 在模板上访问时会 自动浅层解包 因此你无须再在模板中为它写 .value。当通过 this 访问时也会同样如此解包。

:::tip setup() 自身并不含对组件实例的访问权,即在 setup() 中访问 this 会是 null。你可以从选项式 API 中访问由组合式 API 暴露的值,但反过来则不行。 :::

访问 Props {#accessing-props}

setup 函数的第一个参数是组件的 props。和标准的组件一致,一个 setup 函数的 props 是响应式的,并且会在传入新的 props 时同步更新。

  1. export default {
  2. props: {
  3. title: String
  4. },
  5. setup(props) {
  6. console.log(props.title)
  7. }
  8. }

注意如果你从 props 对象上解构,被解构的变量将会丢失响应性。因此我们推荐通过 props.xxx 的形式来使用其中的属性。

如果你确实需要从 props 上解构,或者想要将某个 prop 传入到一个外部函数中但想保持响应性,那么你可以使用 toRefs() toRef() 这两个工具 API:

  1. import { toRefs } from 'vue'
  2. export default {
  3. setup(props) {
  4. // 将 `props` 转为一个其中全是 ref 的对象,然后解构
  5. const { title } = toRefs(props)
  6. // `title` 是一个追踪着 `props.title` 的 ref
  7. console.log(title.value)
  8. // 或者,将 `props` 的单个属性转为一个 ref
  9. const title = toRef(props, 'title')
  10. }
  11. }

Setup 的上下文 {#setup-context}

传入 setup 函数的第二个参数是一个 Setup 上下文 对象。上下文对象上暴露了其他一些在 setup 之中很有用的值:

  1. export default {
  2. setup(props, context) {
  3. // Attributes(不是响应式的对象,等价于 $attrs)
  4. console.log(context.attrs)
  5. // 插槽(不是响应式的对象,等价于 $slots)
  6. console.log(context.slots)
  7. // 抛出事件(函数,等价于 $emit)
  8. console.log(context.emit)
  9. // 暴露公共属性(函数)
  10. console.log(context.expose)
  11. }
  12. }

该上下文对象是非响应式的,可以安全地解构:

  1. export default {
  2. setup(props, { attrs, slots, emit, expose }) {
  3. ...
  4. }
  5. }

attrsslots 都是有状态的对象,它们总会随着组件自身的变更而更新。这意味着你应当避免从它们之中解构,而是始终使用属性引用,例如 attrs.xslots.x 这样。此外还需注意,和 props 不同,attrsslots 上的属性都 不是 响应式的。如果你试图基于 attrsslots 来应用副作用,则应该放在生命周期钩子 onBeforeUpdate 之中。

暴露公共属性 {#exposing-public-properties}

expose 这个函数可以用于在父组件中通过模板 ref访问本组件时,显式地限制所暴露的属性:

  1. export default {
  2. setup(props, { expose }) {
  3. // 这样会使得该组件处于 “关闭状态”
  4. // 即不向父组件暴露任何东西
  5. expose()
  6. const publicCount = ref(0)
  7. const privateCount = ref(0)
  8. // 有选择地暴露局部状态
  9. expose({ count: publicCount })
  10. }
  11. }

渲染函数的用法 {#usage-with-render-functions}

setup 也可以返回一个 渲染函数,它可以直接使用在同一作用域中声明的响应状态:

  1. import { h, ref } from 'vue'
  2. export default {
  3. setup() {
  4. const count = ref(0)
  5. return () => h('div', count.value)
  6. }
  7. }

返回一个渲染函数将会阻止我们返回其他东西。对组件内部来说,这应该不是个问题,但如果我们想通过模板 ref 将这个组件的方法暴露给父组件,那就有问题了。

我们可以通过调用 expose() 解决这个问题:

  1. import { h, ref } from 'vue'
  2. export default {
  3. setup(props, { expose }) {
  4. const count = ref(0)
  5. const increment = () => ++count.value
  6. expose({
  7. increment
  8. })
  9. return () => h('div', count.value)
  10. }
  11. }

此时这个 increment 方法将可以在父组件中模板 ref 上访问到。