【Vue3】依赖注入

provide 和 inject 是 Vue.js 中用于实现依赖注入的两个关联功能。它们允许你在祖先组件中提供数据,然后在子孙组件中注入这些数据,实现组件之间的数据共享和传递。

  • provide:provide 是一个选项,你可以在父组件中通过它来提供数据。你可以在 provide 选项中声明要共享的数据。这些数据可以是任何 JavaScript 值,包括对象、函数等。

  • inject:inject 是另一个选项,你可以在子孙组件中通过它来注入父组件提供的数据。inject 的值是一个包含了你想要注入的属性名称的数组或对象。

App.vue

<template>
  <h1>App.vue(爷爷级别)</h1>
  <label>
    <input v-model="colorVal" value="red" name="color" type="radio">红色
  </label>
  <label>
      <input v-model="colorVal" value="pink" name="color" type="radio">粉色
    </label>
    <label>
      <input v-model="colorVal" value="yellow" name="color" type="radio">黄色
    </label>

    <div class="box">

    </div>
    <hr>
    <provideAVue></provideAVue>
</template>

<script setup lang="ts">
import {ref, reactive, inject, readonly,provide} from 'vue'
import provideAVue from './components/provideA.vue'
const colorVal = ref<string>('red')
provide('color', colorVal)
// 为防止子组件修改color的值,可以用readonly绑定 provide('color', readonly(colorVal))
</script>

<style scoped>
.box {
  height: 50px;
  width: 50px;
  border: 1px solid #ccc;
  background: v-bind(colorVal);
}
</style>

provideA.vue

<template>
  <h1>provideA.vue(父亲级别)</h1>
  <div class="box">

  </div>
  <hr>
  <provideBVue></provideBVue>
</template>

<script setup lang="ts">
import { ref, reactive, inject } from 'vue';
// 直接<ref<string>>会报错:'ref' refers to a value, but is being used as a type here. Did you mean 'typeof ref'?  需要用<Ref<string>> 引入 type {Ref}
import type {Ref} from 'vue'
import provideBVue from './provideB.vue'
const color = inject<Ref<string>>('color')

</script>

<style scoped>
.box {
  height: 50px;
  width: 50px;
  border: 1px solid #ccc;
  background: v-bind(color);
}
</style>

provideB.vue

<template>
  <h1>provideB.vue(孙子级别)</h1>
  <div>
    <button @click="change">修改 provide 的值 yellow</button>
  </div>
  <div class="box">

  </div>
  <hr>
  <provideBVue></provideBVue>
</template>

<script setup lang="ts">
import { ref, reactive, inject } from 'vue';
import type { Ref } from 'vue'
const color = inject<Ref<string>>('color', ref('red'))
// provide 传过来的值可以被子组件修改,不想被修改可以使用 readonly 对变量进行绑定
const change = () => {
  // 报错:'color' is possibly 'undefined'.  
  // color.value = 'yellow'

  // 报错:The left-hand side of an assignment expression may not be an optional property access.
  // color?.value = 'yellow'

  // 解决方法一:使用 !. 非空断言解决
  // color!.value = 'yellow'

  // 解决方法二:添加默认值, ref('red')
  color.value = 'yellow'
}

</script>

<style scoped>
.box {
  height: 50px;
  width: 50px;
  border: 1px solid #ccc;
  background: v-bind(color);
}
</style>

需要注意的是,provide 和 inject 不适用于跨级的父子组件关系,因为 Vue 的设计是鼓励将组件树限制在父子关系中,而不是过多地嵌套。在跨级关系中,你可能需要使用 Vuex 或事件总线等其他方法来实现数据共享。

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

小秀_heo

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值