组件通信-provide和inject

跨层级通信-provide和inject

当组件嵌套层级很深时,使用 Props 层层传递非常痛苦。这时可以使用 provideinject

我们之前学过,祖孙通信也可以通过$attrs,但是那不是直接的,它还需要通过中间子组件来进行绑定;而这次学习可以直接越过子组件,祖孙之间直接通信;

  • 我们首先在爷爷组件上定义一些数据
vue 复制代码
<template>
  <div class="box">
    <h3>父组件</h3>
    <h4>存款:{{ money }}万元</h4>
    <h4>一辆{{ car.brand }}车,价值{{ car.price }}万元</h4>
    <Child />
  </div>
</template>

<script setup lang="ts">
import Child from './Child.vue'

import { ref, reactive } from 'vue';
let money = ref(100);
let car = reactive({
  brand: '宝马',
  price: 100
})
</script>
  • 然后我们就可以使用provide像孙组件提供数据了,provide接受两个参数,第一个参数式名字,第二个是值
vue 复制代码
import { ref, reactive, provide } from 'vue';
//向后代组件提供数据
provide('money', money)
provide('car', car)
  • 然后我们在孙组件中使用inject注入祖先组件给我提供的数据即可
vue 复制代码
<template>
    <div class="box">
        <h3>孙组件</h3>
        <h4>{{ money }}</h4>
        <h4>{{ car.brand }},{{ car.price }}</h4>
    </div>
</template>

<script setup lang="ts">
import { inject } from 'vue';

let money = inject('money')
let car = inject('car')
</script>
  • 这里我们会发现这里会有报错
  • 这个报错提示很明显,告诉我们car对象无法推断出它的类型;解决这个麻烦我们可以使用inject的第二个参数,inject的第二个参数是默认值,意思就是如果祖先组件给我们提供的数据突然失效了,那我们就使用我们自己的定义的默认值;
vue 复制代码
<template>
    <div class="box">
        <h3>孙组件</h3>
        <h4>{{ money }}</h4>
        <h4>{{ car.brand }},{{ car.price }}</h4>
    </div>
</template>

<script setup lang="ts">
import { inject } from 'vue';

let money = inject('money', '未知')
let car = inject('car', { brand: '未知', price: '未知' })
</script>

provide/inject只能向下传递,如果我们向要向上传递的话,就需要祖先组件提供一个方法给后代组件来实现向上传递

  • 例如我们现在定义一更新money的函数
vue 复制代码
function updateMoney(value: number) {
  money.value - + value

}

//像后代组件提供数据
provide('money', { money, updateMoney })
provide('car', car)
  • 一样的,我们还是需要在后代组件中进行注入
vue 复制代码
let { money, updateMoney } = inject('money', { money: 0, updateMoney: (param: number) => { } })

注意这里也要写默认值,不然ts仍然会报错

vue 复制代码
        <button @click="updateMoney(10)">点击拿爷爷的钱</button>

provide/inject 的原理

provide和inject是基于原型链实现的

  • 当创建一个组件实例的时候,provides对象默认会指向父组件的provides对象;
  • 如果这个组件调用了provide,Vue会以父组件provides对象为原型创建一个新的对象,然后将新提供的键值(就是名字和值)添加到这个新的对象上,并复制给当前组件的provides;
  • 所以,当后代在查找注入的时候,会沿着自己的provides对象开始,顺着原型链一路向上查找,直到找到符合的key或者一直到顶端;
相关推荐
candyTong1 天前
一觉醒来,大模型就帮我排查完页面性能问题
前端·javascript·架构
玩嵌入式的菜鸡1 天前
网页访问单片机设备---基于mqtt
前端·javascript·css
前端一小卒1 天前
我用 Claude Code 的 Superpowers 技能链写了个服务,部署前差点把服务器搞炸
前端·javascript·后端
阿丰资源1 天前
SpringBoot+Vue实战:打造企业级在线文档管理系统
vue.js·spring boot·后端
豹哥学前端1 天前
用猜数字游戏,一口气掌握 JavaScript 核心知识点(附完整代码)
前端·javascript
忆往wu前1 天前
从0到1一步步拆解搭建,梳理一个 Vue3 简易图书后台全开发流程
前端·javascript·vue.js
shao9185161 天前
第3章(2)——使用Gradio JavaScript Client
javascript·node.js·cdn·gradio·job·events·playcode
光影少年1 天前
大屏页面,一次多个请求,请求加密导致 点击 全局时间选择器 时出现卡顿咋解决(面板收起会延迟1~2秒)
前端·javascript·vue.js·学习·前端框架·echarts·reactjs
Mr.mjw1 天前
vue中封装一个环形进度条组件,根据外部盒子大小自适应变化
前端·javascript·vue.js
无心使然1 天前
Openlayers调用ArcGis影像服务之一动态地图、地图切片(/exportImage)
前端·javascript·数据可视化