Bootstrap

vue祖孙组件通信传值 provide 与 inject 以及 数据的响应式

vue祖孙组件通信传值 provide 与 inject 以及 数据的响应式

通常,当我们需要从父组件向子组件传递数据时,我们使用 props。但是对于一些深度嵌套的组件,深层的子组件只需要父组件的部分内容。在这种情况下,如果仍然将 prop 沿着组件链逐级传递下去,可能会很麻烦。

provide 和 inject无论组件层次结构有多深,父组件都可以作为其所有子组件的依赖提供者。解决了跨层级传递属性的不方便。

一、概念解析

  • 成对出现:provide和inject是成对出现的
  • 作用:用于父组件向子孙组件传递数据
  • 使用方法:
    • provide在父组件中返回要传给下级的数据;
    • inject在需要使用这个数据的子辈组件或者孙辈等下级组件中注入数据。
  • 使用场景:由于vue有$parent属性可以让子组件访问父组件。但孙组件想要访问祖先组件就比较困难。通过provide/inject可以轻松实现跨级访问父组件的数据

二、用法解析

1 provide

在 祖辈组件的provide 中指定要传递给子孙组件的数据。

data() {
    return {     
        data:"给子孙的数据"   
    } 
},
provide() {   
    return {    
        data: this.data   
    } 
},

2 inject

子孙组件通过inject注入祖父组件传递过来的数据。

// inject用来指定一个数组或者一个对象,数组的话就放provide里字段的名称,而对象的话可以指定
data() {
  return {}
},
inject: ['data'],


inject: {
    data: {
      from: 'data',
      default: ""
    }
  }

但是此时data并不是响应式的数据.....

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

provide和inject数据响应式的解决方案

一、把值转为函数,记得要用箭头函数,不然不能正确获取this

data () {
    return {
        data: ""
    }
}
provide() {
  return {
    data: () => {
      return this.data;
    },
  };}

然后使用时就要变成了函数的调用

inject: ['data'],
computed: {
    data1() {
        return this.data()
    }
}

此时当祖辈组件的data改变时, 子组件中data1的值也会改变

二、把provide所在的Vue实例给传递下去

data () {
    return {
        data: ""
    }
}
provide() {
  return {
    ParentCom: this,
  };
  }

在子组件使用时:

inject: ['ParentCom'],
methods:{
    data_func () {
      console.log(ParentCom.data)
    }
}

之所以官方文档对此特性语焉不详,就是因为从他们的出发角度看不是特别推荐使用它,同时也怕开发者滥用。而从业务开发的角度来讲,绝大部分场景应该选用Vuex而不是provide/inject。 Vuex背后是原型链和响应式,并没有用到provide/inject。

小心内存泄漏!

这个问题是最近在实际开发中发现的,其实正中了官方所担心的使用不恰当的问题。
起因是provide里提供了一个方法,在子组件里调用这个方法,把子组件里的图表实例传递进去Hold住,以便将来调用。这样一来,图表的实例相当于一直存在于内存里而无法被回收掉,没过多久页面就卡顿的不行了。

;