您好,欢迎来到三六零分类信息网!老站,搜索引擎当天收录,欢迎发信息

Vue组件通信的六种方式

2025/9/4 20:17:07发布22次查看
在平时的开发过程中,父子 / 兄弟组件间的通信是肯定会遇到的啦,所以这里总结了 6 种 vue 组件的通信props / $e$emit / vuex$attrs / $listeners$parent / $children 与 ref
provide / inject
前言  
如上图所示,a/b,b/c,b/d 组件是父子关系,c/d 是兄弟关系。那如何根据不同的使用场景,选择不同的通信方式呢?所以前提就是我们要了解不同的通信方式的作用和区别。
一. props / $emit这个是我们平时用得比较多的方式之一,父组件 a 通过 props 参数向子组件 b 传递数据,b 组件通过 $emit 向 a 组件发送一个事件(携带参数数据),a组件中监听 $emit 触发的事件得到 b 向 a 发送的数据。 我们来具体解释下它的实现步骤:
1:父组件向子组件传值// app.vue 父组件<template>    <a-compontent :data-a="dataa"></a-compontent></template><script>import acompontent from './components/a.vue';export default {    name: 'app',    compontent: { acompontent },        data () {                return {         dataa: 'dataa数据'       }    }}// acompontent 子组件<template>    <p>{{dataa}}</p> // 在子组件中把父组件传递过来的值显示出来</template><script>export default {    name: 'acompontent',    props: {        dataa: {                         //这个就是父组件中子标签自定义名字                          type: string,            required: true  // 或者false        }    }}</script>
2:子组件向父组件传值(通过事件方式)// 子组件<template>    <p @click="senddatatoparent">点击向父组件传递数据</p></template><script>export default {    name: 'child',    methods:{                changetitle() {              // 自定义事件,会触发父组件的监听事件,并将数据以参数的形式传递            this.$emit('senddatatoparent','这是子组件向父组件传递的数据');         }    }}// 父组件<template>    <child @senddatatoparent="getchilddata"></child></template><script>import child from './components/child.vue';    export default {    name: 'child',    methods:{        getchilddata(data) {            console.log(data); // 这里的得到了子组件的值        }    }}</script>
二. $emit / $on这种方式是通过一个类似 app.vue 的实例作为一个模块的事件中心,用它来触发和监听事件,如果把它放在 app.vue 中,就可以很好的实现任何组件中的通信,但是这种方法在项目比较大的时候不太好维护。
举个: 假设现在有 4 个组件,home.vue 和 a/b/c 组件,ab 这三个组件是兄弟组件,home.vue 相当于父组件 建立一个空的 vue 实例,将通信事件挂载在该实例上 -
d.jsimport vue from 'vue'export default new vue()
// 我们可以在router-view中监听change事件,也可以在mounted方法中监听// home.vue<template>  <p>    <child-a />    <child-b />    <child-c />  </p></template>
// a组件<template>  <p @click="dataa">将a组件的数据发送给c组件 - {{name}}</p></template><script>import event from ./d;export default {  data() {    return {      name: 'echo'    }  },  components: { event },  methods: {    dataa() {      event.$emit('data-a', this.name);    }  }}</script>
// b组件<template>  <p @click="datab">将b组件的数据发送给c组件 - {{age}}</p></template><script>import event from ./d;export default {  data() {    return {      age: '18'    }  },  components: { event },  methods: {    datab() {      event.$emit('data-b', this.age);    }  }}</script>
// c组件<template>  <p>c组件得到的数据 {{name}} {{age}}</p></template><script>import event from ./d;export default {  data() {    return {      name: '',      age: ''    }  },  components: { event },  mounted() {    // 在模板编译完成后执行    event.$on('data-a', name => {      this.name = name;    })    event.$on('data-b', age => {      this.age = age;    })  }}</script>
上面的里我们可以知道,在 c 组件的 mounted 事件中监听了 a/b 的 $emit 事件,并获取了它传递过来的参数(由于不确定事件什么时候触发,所以一般在 mounted / created 中监听)
三. vuexvuex 是一个状态管理模式。它采用集中式存储管理应用的所有组件的状态,并以相应的规则保证状态以一种可预测的方式发生变化。 vuex 应用的核心是 store(仓库,一个容器),store 包含着你的应用中大部分的状态 (state);
这个部分就不详细介绍了,官方文档很详细了 vuex.vuejs.org/zh/guide/st…
四. $attrs / $listeners
如上图所示,这是一个多级组件的嵌套,那 a/c 组件如何进行通信?我们现在可以想到的有下面几种方案:
使用 vuex 来进行数据管理,但是使用的 vuex 的问题在于,如果项目比较小,组件间的共享状态比较少,那用 vuex 就好比杀鸡用牛刀。利用 b 组件做中转站,当 a 组件需要把信息传给 c 组件时,b 接受 a 组件的信息,然后用 props 传给 c 组件, 但是如果嵌套的组件过多,会导致代码繁琐,代码维护比较困难;如果 c 中状态的改变需要传递给 a, 还要使用事件系统一级级往上传递 。在 vue2.4 中,为了解决该需求,引入了attrs和listeners , 新增了inheritattrs 选项。(如下图所示)
$attrs 的作用,某些情况下需要结合 inheritattrs 一起使用
有 4 个组件:app.vue / child1.vue / child2.vue / child3.vue,这 4 个组件分别的依次嵌套的关系。
// app.vue<template>  <p id="app">    <p>app.vue</p><hr>    // 这里我们可以看到,app.vue向下一集的child1组件传递了5个参数,分别是name / age / job / sayhi / title    <child1 :name="name" :age="age" :job="job" :say-hi="say" title="app.vue的title"></child1>  </p></template><script>const child1 = () => import(./components/child1.vue);export default {  name: 'app',  components: { child1 },  data() {    return {      name: echo,      age: 18,      job: fe,      say: this is hi~    };  }};</script>
// child1.vue<template>  <p class="child1">    <p>child1.vue</p>    <p>name: {{ name }}</p>    <p>childcom1的$attrs: {{ $attrs }}</p>    <p>可以看到,$attrs这个对象集合中的值 = 所有传值过来的参数 - props中显示定义的参数</p>    <hr>    <child2 v-bind="$attrs"></child2>  </p></template><script>const child2 = () => import(./child2.vue);export default {  components: {    child2  },  // 这个inheritattrs默认值为true,不定义这个参数值就是true,可手动设置为false  // inheritattrs的意义在用,可以在从父组件获得参数的子组件根节点上,将所有的$attrs以dom属性的方式显示  inheritattrs: true, // 可以关闭自动挂载到组件根元素上的没有在props声明的属性  props: {    name: string // name作为props属性绑定  },  created() {    // 这里的$attrs就是所有从父组件传递过来的所有参数 然后 除去props中显式定义的参数后剩下的所有参数!!!    console.log(this.$attrs); //  输出{age: 18, job: fe, say-hi: this is hi~, title: app.vue的title}  }};</script>
// child2.vue<template>  <p class="child2">    <p>child2.vue</p>    <p>age: {{ age }}</p>    <p>childcom2: {{ $attrs }}</p>    <hr>    <child3 v-bind="$attrs"></child3>  </p></template><script>const child3 = () => import(./child3.vue);export default {  components: {    child3  },  // 将inheritattrs设置为false之后,将关闭自动挂载到组件根元素上的没有在props声明的属性  inheritattrs: false,  props: {    age: string  },  created() {    // 同理和上面一样,$attrs这个对象集合中的值 = 所有传值过来的参数 - props中显示定义的参数    console.log(this.$attrs);  }};</script>
// child3.vue<template>  <p class="child3">    <p>child3.vue</p>    <p>job: {{job}}</p>    <p>title: {{title}}</p>    <p>childcom3: {{ $attrs }}</p>  </p></template><script>export default {  inheritattrs: true,  props: {    job: string,    title: string  }};</script>
来看下具体的显示效果:
而$listeners怎么用呢,官方文档说的是:包含了父作用域中的 (不含 .native 修饰器的) v-on 事件监听器。它可以通过 v-on=$listeners 传入内部组件——在创建更高层次的组件时非常有用! 从字面意思来理解应该是在需要接受值的父组件增加一个监听事件?话不多说,上代码
还是 3 个依次嵌套的组件
<template>  <p class="child1">    <child2 v-on:uprocket="reciverocket"></child2>  </p></template><script>const child2 = () => import(./child2.vue);export default {  components: {    child2  },  methods: {    reciverocket() {      console.log(reciverocket success);    }  }};</script>复制代码
<template>  <p class="child2">    <child3 v-bind="$attrs" v-on="$listeners"></child3>  </p></template><script>const child3 = () => import(./child3.vue);export default {  components: {    child3  },  created() {    this.$emit('child2', 'child2-data');  }};</script>复制代码
// child3.vue<template>  <p class="child3">    <p @click="startuprocket">child3</p>  </p></template><script>export default {  methods: {    startuprocket() {      this.$emit(uprocket);      console.log(startuprocket);    }  }};</script>复制代码
这里的结果是,当我们点击 child3 组件的 child3 文字,触发 startuprocket 事件,child1 组件就可以接收到,并触发 reciverocket 打印结果如下:
> reciverocket success> startuprocket
五. $parent / $children 与 refref:如果在普通的 dom 元素上使用,引用指向的就是 dom 元素;如果用在子组件上,引用就指向组件实例$parent / $children:访问父 / 子实例这两种方式都是直接得到组件实例,使用后可以直接调用组件的方法或访问数据。
我们先来看个用 ref 来访问组件的:
// child1子组件export default {          data() {                return {      title: 'vue.js'    };  },  methods: {          sayhello() {      console.log('child1!!');    }  }};
// 父组件<template>  <child1 @click="sayhi" ref="child1"></child1></template><script>  export default {    methods: {              sayhi () {        const child1 = this.$refs.child1;        console.log(child1.title);  // vue.js        child1.sayhello();  // 弹窗      }    }  }</script>
六. provide/injectprovide/inject 是 vue2.2.0 新增 api,这对选项需要一起使用,以允许一个祖先组件向其所有子孙后代注入一个依赖,不论组件层次有多深,并在起上下游关系成立的时间里始终生效。如果你熟悉 react,这与 react 的上下文特性很相似。
provide 和 inject 主要为高阶插件/组件库提供用例。并不推荐直接用于应用程序代码中。
由于自己对这部分的内容理解不是很深刻,所以感兴趣的可以前往官方文档查看: cn.vuejs.org/v2/api/#pro…
总结常见使用场景可以分为三类:
父子通信:props / $emit;$parent / $children;$attrs/$listeners;provide / inject api; ref兄弟通信:vuex跨级通信:vuex;$attrs/$listeners;provide / inject api
4.接下来我还会在我的裙里用视频讲解的方式给大家讲解【以下图中的内容】有兴趣的可以来我的扣扣裙 519293536 免费交流学习,我都会尽力帮大家哦
推荐教程:《js教程》
以上就是vue组件通信的六种方式的详细内容。
该用户其它信息

VIP推荐

免费发布信息,免费发布B2B信息网站平台 - 三六零分类信息网 沪ICP备09012988号-2
企业名录 Product