组件通信-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或者一直到顶端;
相关推荐
万少20 小时前
万少的博客 - 技术分享与解决方案
前端·javascript·后端
尘世中一位迷途小书童1 天前
用 Cesium 撸了一个森林火情监控大屏,弧线、粒子、发光效果都齐了
前端·javascript
先吃饱再说1 天前
JavaScript中`this` 的“千层套路”:从默认绑定到箭头函数的五种指向
javascript
foxire1 天前
基于nodejs实现服务端内核引擎
javascript
锋行天下1 天前
半秒开!还有谁!!!
前端·vue.js·架构
触底反弹1 天前
🧠 搞懂 Token,才算真正入门大模型——从分词原理到 Embedding 语义实战
javascript·人工智能·算法
free351 天前
AST Interpreter 的设计:为什么分 evaluate() 和 execute()
javascript
JING小白1 天前
Day 1 重学Vue:响应式系统的“底层逻辑”变更,Vue2旧时代的终结与Vue3新时代的开启
前端·vue.js
等咸鱼的狸猫1 天前
JavaScript 隐式类型转换:从入门到精通
javascript
kyriewen1 天前
我用 Codex 重写了同事维护三年的代码,他没说谢谢——而是找了领导
前端·javascript·ai编程