provide/inject

本文深入探讨Vue.js中provide/inject的使用方法,解析如何在组件间传递数据,包括基本用法、监听数据变化及刷新组件技巧。适用于Vue开发者理解和掌握高级组件通信方式。

概述

provide/inject:简单的来说就是在父组件中通过provide来提供变量,然后在子组件中通过inject来注入变量。

需要注意的是这里不论子组件有多深,只要调用了inject那么就可以注入provide中的数据。而不是局限于只能从当前父组件的prop属性来获取数据。

写法

// 传入对象写法
provide: {
    foo: 'bar'
}
// 函数写法
provide () {
    return {
      foo: 'bar'
    }
}

实例

祖孙传参例子:
//Parent:
<template>
 <div>
<childOne></childOne>
 </div>
</template>
 
<script>
 import childOne from '../components/test/ChildOne'
 export default {
  name: "Parent",
  provide: {
   for: "demo"
  },
  components:{
   childOne
  }
 }
//ChildOne:
<template>
 <div>
  {{demo}}
  <childtwo></childtwo>
 </div>
</template>
 
<script>
 import childtwo from './ChildTwo'
 export default {
  name: "childOne",
  inject: ['for'],
  data() {
   return {
    demo: this.for
   }
  },
  components: {
   childtwo
  }
 }
</script>
//ChildTwo:
<template>
 <div>
  {{demo}}
 </div>
</template>
 
<script>
 export default {
  name: "",
  inject: ['for'],
  data() {
   return {
    demo: this.for
   }
  }
 }
</script>

在这里插入图片描述

实现 provide / inject 可监听
// App.vue 组件 (父组件)
export default {
        name: 'electron-xdf',
        data () {
            return {
                asideW: '200px',
                _path: null
            };
        },
        created () {
            this._path = this.$route.path;
        },
        provide() {
            return {
                INFO: this // 传递可响应对象
            };
        },
        methods:{
            upprovide(){
                this.asideW = '500px';  // 修改 属性值得变化,可以传递到 子孙后代的任意组件 同时响应变化
            }
        }
    }
// 子组件
export default {
        props: {},
        computed: {},
        name: 'process-communication',
        data () {
            return {
                count: 0
            };
        },
        inject: ['INFO'],
        methods: {
            upprovide(){
                // 这里修改之后 App.vue 也会响应数据的变化
                this.INFO.asideW = '200px';
            }
        }
    };

提示:provide 和 inject 绑定并不是可响应的。这是刻意为之的。然而,如果你传入了一个可监听的对象,那么其对象的属性还是可响应的。

刷新vue组件
// App.vue 组件 (父组件)
<template>
  <div
    id="app"
  >
    <router-view
      v-if="isRouterAlive"
    />
  </div>
</template>

<script>
export default {
  name: 'App',
  components: {
    MergeTipDialog,
    BreakNetTip
  },
  data () {
    return {
      isShow: false,
      isRouterAlive: true
  },

// 父组件中返回要传给下级的数据
  provide () {
    return {
      reload: this.reload
    }
  },
  methods: {
    reload () {
      this.isRouterAlive = false
      this.$nextTick(() => {
        this.isRouterAlive = true
      })
    }
  }
}
</script>
// 子组件
<template>
  <popup-assign
    :id="id"
    @success="successHandle"
  >
    <div class="confirm-d-tit"><span class="gray-small-btn">{{ name }}</span></div>
    <strong>将被分配给</strong>
    <a
      slot="reference"
      class="unite-btn"
    >
      指派
    </a>
  </popup-assign>
</template>
<script>
import PopupAssign from '../PopupAssign'
export default {
//引用vue reload方法
  inject: ['reload'],
  components: {
    PopupAssign
  },
methods: {
    // ...mapActions(['freshList']),
    async successHandle () {
      this.reload()
    }
  }
}
</script>

这样就实现了子组件调取reload方法就实现了刷新vue组件的功能,个人认为它实现了组件跨越组件传递数据方法。

参考:
https://www.jb51.net/article/139769.htm
https://blog.youkuaiyun.com/qq_15253407/article/details/91490210
https://www.jianshu.com/p/d34a7df4cd6a

### ProvideInject 的概念 ProvideInjectVue 3 实现依赖注入的关键机制。父组件能够通过 `provide` 提供数据,而任何后代组件则可以通过 `inject` 获取这些数据[^1]。 ### Provide 方法详解 `provide` 主要用于父级组件向其所有子孙组件提供共享的数据或功能。这使得在整个子树内传播信息变得更加容易,尤其是在处理全局配置项或是需要广泛访问的状态时非常有用。为了使提供的属性具有响应性,应当利用 `ref()` 或者 `reactive()` 来创建响应式对象[^3]。 ```javascript // 父组件 ParentComponent.vue 使用 setup() 函数的方式 import { provide, ref } from 'vue'; export default { name: "ParentComponent", setup() { const title = ref("默认标题"); // 将响应式变量作为可注入的内容暴露出去 provide('title', title); return {}; } } ``` ### Inject 方法详解 相对地,`inject` 则是在那些希望接收到由祖先节点所提供的资源的地方调用的方法。它可以是一个字符串名称列表或者是带有选项的对象形式指定所需注入的键名及其对应的解析逻辑。对于非响应性的简单值来说,一旦被消费端捕获之后就不会再随源变化而同步更新;而对于基于 `ref`/`reactive` 构建起来的服务实例,则可以在整个生命周期里保持最新状态并触发视图重绘[^4]。 ```javascript // 子组件 ChildComponent.vue 接收来自上级提供的数据 import { inject } from 'vue'; export default { name: "ChildComponent", setup() { // 注入名为'title'的响应式引用 const title = inject('title'); return { title }; }, template: `<div>{{ title }}</div>` }; ``` ### Provide vs Inject 的主要差异 | 特征 | Provide | Inject | |--| | **角色定位** | 负责向外发布可供下游使用的公共资产 | 用来声明当前上下文中期望获得哪些外部供给 | | **作用范围** | 可影响到所有的直系及间接下层组件 | 仅限于自身以及更深层嵌套的后代 | | **传递内容** | 支持任意 JavaScript 值 | 同样支持各种 JS 类型 | | **响应特性** | 需显式转换为 Ref/reactive 才能维持动态效果 | 自动继承自上游提供的响应行为 |
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值