vue组件间通信六种方式(总结篇)(3)

// demo.vue <template> <div> <child-com:foo="foo":boo="boo":coo="coo":doo="doo"></child-com> </div> </tempalte> <script> const childCom = ()=> import('./childCom1.vue') export default { data () { return { foo: 'Hello World!', boo: 'Hello Javascript!', coo: 'Hello Vue', doo: 'Last' } }, components: { childCom } } </script> // childCom1.vue <template> <div> <p>foo: {{ foo }}</p> <p>attrs: {{ $attrs }}</p> <child-com2 v-bind="$attrs"></child-com2> </div> </template> <script> const childCom2 = ()=> import('./childCom2.vue') export default { props: ['foo'], // foo作为props属性绑定 inheritAttrs: false, created () { console.log(this.$attrs) // { boo: 'Hello Javascript!', coo: 'Hello Vue', doo: 'Last' } } } </script> // childCom2.vue <template> <div> <p>boo: {{ boo }}</p> <p>attrs: {{ $attrs }}</p> <child-com3 v-bind="$attrs"></child-com3> </div> </template> <script> const childCom3 = ()=> import('./childCom3.vue') export default { props: ['boo'] // boo作为props属性绑定 inheritAttrs: false, created () { console.log(this.$attrs) // { coo: 'Hello Vue', doo: 'Last' } } } </script>

$attrs 表示没有继承数据的对象,格式为{属性名:属性值}。Vue2.4提供了 $attrs  ,  $listeners  来传递数据与事件,跨级组件之间的通讯变得更简单

方法五、provide/inject

1.简介

Vue2.2.0新增API,这对选项需要一起使用, 以允许一个祖先组件向其所有子孙后代注入一个依赖,不论组件层次有多深,并在起上下游关系成立的时间里始终生效 。一言而蔽之: 祖先组件中通过provider来提供变量,然后在子孙组件中通过inject来注入变量。

2.举个例子

假设有两个组件: A.vue 和 B.vue,B 是 A 的子组件

// A.vue export default { provide: { name: '浪里行舟' } } // B.vue export default { inject: ['name'], mounted () { console.log(this.name); // 浪里行舟 } }

可以看到,在 A.vue 里,我们设置了一个  provide: name ,值为 浪里行舟,它的作用就是将  name  这个变量提供给它的所有子组件。而在 B.vue 中,通过  inject  注入了从 A 组件中提供的  name  变量,那么在组件 B 中,就可以直接通过  this.name  访问这个变量了,它的值也是 浪里行舟。这就是 provide / inject API 最核心的用法。

需要注意的是: provide 和 inject 绑定并不是可响应的 。不过,如果你传入了一个可监听的对象,那么其对象的属性还是可响应的。   所以,上面 A.vue 的 name 如果改变了,B.vue 的 this.name 是不会改变的,仍然是 浪里行舟。

方法六、 $parent  /  $children 与  ref

ref :如果在普通的 DOM 元素上使用,引用指向的就是 DOM 元素;如果用在子组件上,引用就指向组件实例

$parent  /  $children :访问父 / 子实例

需要注意的是:这两种都是直接得到组件实例,使用后可以直接调用组件的方法或访问数据。我们先来看个用  ref 来访问组件的例子:

// component-a 子组件 export default { data () { return { title: 'Vue.js' } }, methods: { sayHello () { window.alert('Hello'); } } } // 父组件 <template> <component-a ref="comA"></component-a> </template> <script> export default { mounted () { const comA = this.$refs.comA; console.log(comA.title); // Vue.js comA.sayHello(); // 弹窗 } } </script>

不过, 这两种方法的弊端是,无法在跨级或兄弟间通信 。

// parent.vue <component-a></component-a> <component-b></component-b> <component-b></component-b>

我们想在 component-a 中,访问到引用它的页面中(这里就是 parent.vue)的两个 component-b 组件,那这种情况下,就得配置额外的插件或工具了,比如 Vuex 和 Bus 的解决方案。

总结

常见使用场景可以分为三类:

父子通信:   父向子传递数据是通过 props,子向父是通过 events( $emit );通过父链 / 子链也可以通信( $parent  /  $children );ref 也可以访问组件实例;provide / inject API。

兄弟通信:   Bus;Vuex;

跨级通信:   Bus;Vuex;provide / inject API、 $attrs/$listeners

内容版权声明:除非注明,否则皆为本站原创文章。

转载注明出处:http://www.heiqu.com/9ce0404e6380418f336a8baca5faa161.html