Vue组件通信的六种方式

在平时的开发过程中,父子 / 兄弟组件间的通信是肯定会遇到的啦,所以这里总结了 6 种 Vue 组件的通信props / $e$emit / Vuex$attrs / $listeners

$parent / $children 与 ref

provide / inject

前言

  61836563bd4c0354d5c80665441803b.png

如上图所示,A/B,B/C,B/D 组件是父子关系,C/D 是兄弟关系。那如何根据不同的使用场景,选择不同的通信方式呢?所以前提就是我们要了解不同的通信方式的作用和区别。

一. props / $emit

这个是我们平时用得比较多的方式之一,父组件 A 通过 props 参数向子组件 B 传递数据,B 组件通过 $emit 向 A 组件发送一个事件(携带参数数据),A组件中监听 $emit 触发的事件得到 B 向 A 发送的数据。 我们来具体解释下它的实现步骤:

立即学习“前端免费学习笔记(深入)”;

1:父组件向子组件传值

// App.vue 父组件    import aCompontent from './components/A.vue';export default {    name: 'app',    compontent: { aCompontent },        data () {                return {         dataA: 'dataA数据'       }    }}// aCompontent 子组件    

{{dataA}}

// 在子组件中把父组件传递过来的值显示出来export default { name: 'aCompontent', props: { dataA: { //这个就是父组件中子标签自定义名字 type: String, required: true // 或者false } }}

登录后复制

2:子组件向父组件传值(通过事件方式)

// 子组件    

点击向父组件传递数据

export default { name: 'child', methods:{ changeTitle() { // 自定义事件,会触发父组件的监听事件,并将数据以参数的形式传递 this.$emit('sendDataToParent','这是子组件向父组件传递的数据'); } }}// 父组件 import child from './components/child.vue'; export default { name: 'child', methods:{ getChildData(data) { console.log(data); // 这里的得到了子组件的值 } }}

登录后复制

二. $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  

登录后复制

// A组件  

将A组件的数据发送给C组件 - {{name}}

import Event from "./D";export default { data() { return { name: 'Echo' } }, components: { Event }, methods: { dataA() { Event.$emit('data-a', this.name); } }}

登录后复制

// B组件  

将B组件的数据发送给C组件 - {{age}}

import Event from "./D";export default { data() { return { age: '18' } }, components: { Event }, methods: { dataB() { Event.$emit('data-b', this.age); } }}

登录后复制

// C组件  

C组件得到的数据 {{name}} {{age}}

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; }) }}

登录后复制

上面的里我们可以知道,在 C 组件的 mounted 事件中监听了 A/B 的 $emit 事件,并获取了它传递过来的参数(由于不确定事件什么时候触发,所以一般在 mounted / created 中监听)

三. Vuex

Vuex 是一个状态管理模式。它采用集中式存储管理应用的所有组件的状态,并以相应的规则保证状态以一种可预测的方式发生变化。 Vuex 应用的核心是 store(仓库,一个容器),store 包含着你的应用中大部分的状态 (state);

这个部分就不详细介绍了,官方文档很详细了 vuex.vuejs.org/zh/guide/st…

四. $attrs / $listeners

Vue组件通信的六种方式

如上图所示,这是一个多级组件的嵌套,那 A/C 组件如何进行通信?我们现在可以想到的有下面几种方案:

使用 Vuex 来进行数据管理,但是使用的 vuex 的问题在于,如果项目比较小,组件间的共享状态比较少,那用 vuex 就好比杀鸡用牛刀。利用 B 组件做中转站,当 A 组件需要把信息传给 C 组件时,B 接受 A 组件的信息,然后用 props 传给 C 组件, 但是如果嵌套的组件过多,会导致代码繁琐,代码维护比较困难;如果 C 中状态的改变需要传递给 A, 还要使用事件系统一级级往上传递 。

在 Vue2.4 中,为了解决该需求,引入了attrs和listeners , 新增了inheritAttrs 选项。(如下图所示)

Vue组件通信的六种方式

 $attrs 的作用,某些情况下需要结合 inheritAttrs 一起使用

有 4 个组件:App.vue / child1.vue / child2.vue / child3.vue,这 4 个组件分别的依次嵌套的关系。

// App.vue  

App.vue


// 这里我们可以看到,app.vue向下一集的child1组件传递了5个参数,分别是name / age / job / sayHi / title

const child1 = () => import("./components/child1.vue");export default { name: 'app', components: { child1 }, data() { return { name: "Echo", age: "18", job: "FE", say: "this is Hi~" }; }};

登录后复制

// child1.vue  

child1.vue

name: {{ name }}

childCom1的$attrs: {{ $attrs }}

可以看到,$attrs这个对象集合中的值 = 所有传值过来的参数 - props中显示定义的参数


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"} }};

登录后复制

// child2.vue  

child2.vue

age: {{ age }}

childCom2: {{ $attrs }}


const child3 = () => import("./child3.vue");export default { components: { child3 }, // 将inheritAttrs设置为false之后,将关闭自动挂载到组件根元素上的没有在props声明的属性 inheritAttrs: false, props: { age: String }, created() { // 同理和上面一样,$attrs这个对象集合中的值 = 所有传值过来的参数 - props中显示定义的参数 console.log(this.$attrs); }};

登录后复制

// child3.vue  

child3.vue

job: {{job}}

title: {{title}}

childCom3: {{ $attrs }}

export default { inheritAttrs: true, props: { job: String, title: String }};

登录后复制

来看下具体的显示效果:

Vue组件通信的六种方式

而$listeners怎么用呢,官方文档说的是:包含了父作用域中的 (不含 .native 修饰器的) v-on 事件监听器。它可以通过 v-on=”$listeners” 传入内部组件——在创建更高层次的组件时非常有用! 从字面意思来理解应该是在需要接受值的父组件增加一个监听事件?话不多说,上代码

还是 3 个依次嵌套的组件

  

const child2 = () => import("./child2.vue");export default { components: { child2 }, methods: { reciveRocket() { console.log("reciveRocket success"); } }};复制代码

登录后复制

  

const child3 = () => import("./child3.vue");export default { components: { child3 }, created() { this.$emit('child2', 'child2-data'); }};复制代码

登录后复制

// child3.vue  

child3

export default { methods: { startUpRocket() { this.$emit("upRocket"); console.log("startUpRocket"); } }};复制代码

登录后复制

这里的结果是,当我们点击 child3 组件的 child3 文字,触发 startUpRocket 事件,child1 组件就可以接收到,并触发 reciveRocket 打印结果如下:

> reciveRocket success> startUpRocket

登录后复制

五. $parent / $children 与 ref

ref:如果在普通的 DOM 元素上使用,引用指向的就是 DOM 元素;如果用在子组件上,引用就指向组件实例$parent / $children:访问父 / 子实例

这两种方式都是直接得到组件实例,使用后可以直接调用组件的方法或访问数据。

我们先来看个用 ref 来访问组件的:

// child1子组件export default {          data() {                return {      title: 'Vue.js'    };  },  methods: {          sayHello() {      console.log('child1!!');    }  }};

登录后复制

// 父组件    export default {    methods: {              sayHi () {        const child1 = this.$refs.child1;        console.log(child1.title);  // Vue.js        child1.sayHello();  // 弹窗      }    }  }

登录后复制

六. provide/inject

provide/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 免费交流学习,我都会尽力帮大家哦

Vue组件通信的六种方式

推荐教程:《JS教程》

以上就是Vue组件通信的六种方式的详细内容,更多请关注【创想鸟】其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至253000106@qq.com举报,一经查实,本站将立刻删除。

发布者:PHP中文网,转转请注明出处:https://www.chuangxiangniao.com/p/2727728.html

(0)
上一篇 2025年3月7日 23:48:15
下一篇 2025年3月7日 23:48:26

AD推荐 黄金广告位招租... 更多推荐

相关推荐

  • 15道Vue常见面试题解析

    1.vue优点? 答: 轻量级框架:只关注视图层,是一个构建数据的视图集合,大小只有几十 kb ; 简单易学:国人开发,中文文档,不存在语言障碍 ,易于理解和学习; 双向数据绑定:保留了 angular 的特点,在数据操作方面更为简单; 组…

    编程技术 2025年3月7日
    200
  • Vue中如何正确强制组件重新渲染?(方法介绍)

    有时候,依赖 Vue 响应方式来更新数据是不够的,相反,我们需要手动重新渲染组件来更新数据。或者,我们可能只想抛开当前的DOM,重新开始。那么,如何让Vue以正确的方式重新呈现组件呢? 强制 Vue 重新渲染组件的最佳方法是在组件上设置:k…

    2025年3月7日
    200
  • Vue中mounted和created的区别(图文详解)

    一、什么是生命周期? 用通俗的语言来说,就是Vue中实例或者组件从创建到消灭中间经过的一系列过程。虽然不太严谨,但是也基本上可以理解。 通过一系列实践,现在把所有遇到的问题整理一遍,今天记录一下created和mounted的区别: 二、c…

    2025年3月7日 编程技术
    200
  • vue中的nexttick原理

    vue中的nexttick原理实现是基于Vue实现响应式并不是数据发生变化之后DOM立即变化,而是按一定的策略进行DOM的更新。 一、原理 1.异步说明 Vue 实现响应式并不是数据发生变化之后 DOM 立即变化,而是按一定的策略进行 DO…

    2025年3月7日
    200
  • 详解vue中watch如何使用?watch用法介绍

    在vue中,使用watch来响应数据的变化。watch的用法大致有三种。下面代码是watch的一种简单的用法: 登录后复制 new Vue({ el: ‘#root’, data: { cityName: ‘shanghai’ }, wat…

    2025年3月7日
    200
  • vue组件通信的中8种方式介绍(收藏)

    vue是数据驱动视图更新的框架, 所以对于vue来说组件间的数据通信非常重要,那么组件之间如何进行数据通信的呢? 首先我们需要知道在vue中组件之间存在什么样的关系, 才更容易理解他们的通信方式。  vue组件中关系说明: 如上图所示, A…

    2025年3月7日
    200
  • javascript之ssm+vue前后端分离框架整合实现

    前言 本文针对Spring+SpringMVC+Mybatis后台开发框架(基于maven构建)与vue前端框架(基于webpack构建)的项目整合进行介绍,对于ssm和vue单独项目的搭建不作为本文的重点,而着重介绍两者之间交互的要点。 …

    2025年3月7日 编程技术
    200
  • vue实现员工信息录入功能的方法

    Vue通用信息录入界面,供大家参考,具体内容如下 nbsp;html>  员工信息录入  .btn1{ color: blue; background: skyblue; text-align: center; }    登录后复制 …

    2025年3月7日
    200
  • 代码详解Vue中key的作用示例

    Vue中key的作用 key的特殊attribute主要用在Vue的虚拟DOM算法,在新旧Nodes对比时辨识VNodes。如果不使用key,Vue会使用一种最大限度减少动态元素并且尽可能的尝试就地修改、复用相同类型元素的算法,而使用key…

    2025年3月7日
    200
  • 12个vue高频原理面试题(附分析)

    本文分享12道高频vue原理面试题,覆盖了 vue 核心实现原理,其实一个框架的实现原理一篇文章是不可能说完的,希望通过这 12 道问题,让读者对自己的 vue 掌握程度有一定的认识(b 数),从而弥补自己的不足,更好的掌握 vue。 【相…

    2025年3月7日 编程技术
    200

发表回复

登录后才能评论