组件通信

Vue组件间通信

一、总结

没错,先总结,后讲解

父子通信

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

兄弟通信

$Bus;Vuex

跨级通信

$Bus;Vuex;provide / inject API、$attrs/$listeners

二、父传子 props

父组件通过v-bind绑定到子组件。子组件通过props接收传过来的值

总结:父组件通过props向下传递数据给子组件。注:组件中的数据共有三种形式:data、props、computed

父组件示例:父子组件关联起来,并通过v-bind(即简写“:”)将父组件中的parentAge值,传递给子组件

v-on绑定的属性名称deliverParentAge与data中定义的parentAge名称可以不一样

属性deliverParentAge通过v-bind绑定的,也是子组件中通过props接收的,而parentAge是要传递给子组件的数值,它是一个值

<template>
    <div class="my-parent">
        <h3>我是父组件,我想告诉我的子组件,我的年龄值是:{{parentAge}}</h3>
        <h3>我要通过v-bind(即简写":")语法糖绑定一个属性deliverParentAge,将父组件的值传递到子组件中</h3>
        <!-- 下面就是我的子组件 -->
        <my-child :deliverParentAge="parentAge"></my-child>

        </div>
    </template>

<script>
import MyChild from "./Child";
export default {
    components: { MyChild },
    data() {
        return {
            parentAge: 49
        };
    }
};
</script>

子组件示例:子组件通过props属性,在子组件中接收父组件传过来的值

<template>
    <div class="my-child">
        <h5>我是子组件,我可以通过属性props来接收父组件传过来的年龄值是:{{deliverParentAge}}</h5>
    </div>
</template>

<script>
export default {
    data() {
        return {
            childAge: 27
        };
    },
    props: {
        deliverParentAge: Number
    }
};
</script>

三、子传父 events($emit)

子组件定义一个事件,父组件通过v-on接收这个事件和传递过来的值

总结:子组件通过events给父组件发送消息,实际上就是子组件把自己的数据发送到父组件。

现在来修改父组件的值(这个不是真的修改而是通过this.$emit提交一个事件,将子组件的行为告诉父组件)

<template>
    <div class="my-child">
        <h5>我是子组件,我可以通过属性props来接收父组件传过来的年龄值是:{{deliverParentAge}},这是一个数字类型</h5>
        <h5>并且,我要告诉他,他今年生日已经过了,所以他的年龄应该<button @click="AddAge">加1</button></h5>
            下面我要通过this.$emit方法提交一个事件addParentAge,告诉我的父组件,他的实际年龄
    </div>
</template>

<script>
export default {
    data() {
        return {
            childAge: 27
        };
    },
    props: {
        deliverParentAge: Number
    },
    // 新建一个计算属性,将父组件原来的值加1
    computed: {
        parentActualAge() {
            return this.deliverParentAge + 1;
        }
    },
    methods: {
        AddAge() {
            this.$emit("addParentAge", this.parentActualAge);
        }
    }
};
</script>

父组件通过语法糖v-on(即简写为“@”)来监听子组件提交的事件addParentAge

this.$emit提交的事件名称addParentAge,与方法handleAddParentAge名称可以不一样

addParentAge是子组件提交的事件名称,也是父组件通过v-on监听的事件名称,而handleAddParentAge是父组件自定义的方法名称

<template>
    <div class="my-parent">
        <h3>我是父组件,我想告诉我的子组件,我的年龄值是:{{parentAge}}</h3>
        <h3>我要通过v-bind(即简写":")语法糖绑定一个属性parentAge,告诉子组件我的年龄值是:{{parentAge}}</h3>
        <!-- 下面就是我的子组件 -->
        <my-child :deliverParentAge="parentAge"
            @addParentAge="handleAddParentAge"></my-child>
        <h3>通过监听子组件提交的事件addParentAge,我知道到了自己的实际年龄应该是:{{parentAge+1}},并通过方法handleAddParentAge,在控制台打印出我的真实年龄</h3>
    </div>
</template>

<script>
import MyChild from "./Child";
export default {
    components: { MyChild },
    data() {
        return {
        parentAge: 49
        };
    },
    methods: {
        handleAddParentAge(actualAge) {
            console.log("父组件的实际年龄是:", actualAge);
        }
    }
};
</script>

四、爷孙通信 provide / inject

简介

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

provide / inject API 主要解决了跨级组件间的通信问题,不过它的使用场景,主要是子组件获取上级组件的状态,跨级组件间建立了一种主动提供与依赖注入的关系。

总结:通过provide传值,通过inject接收。

五、跨级通信 vuex

vuex的原理

Vuex实现了一个单向数据流,在全局拥有一个State存放数据,当组件要更改State中的数据时,必须通过Mutations进行,Mutations同时提供了订阅者模式供外部插件调用获取State数据的更新。而当所有异步操作(常见于调用后端接口异步获取更新数据)或批量的同步操作需要走Action,但Action也是无法直接修改State的,还是需要通过Mutation来修改State的数据。最后,根据State的变化,渲染到视图上。

各模块在流程中的功能

  • Vue Components:Vue组件。HTML页面上,负责接收用户操作等交互行为,执行dispatch方法触发对应action进行回应。
  • dispatch:操作行为触发方法,是唯一能执行action的方法。
  • actions:操作行为处理模块,由组件中的$store.dispatch('action 名称', data1)来触发。然后由commit()来触发mutation的调用 , 间接更新 state。负责处理Vue Components接收到的所有交互行为。包含同步/异步操作,支持多个同名方法,按照注册的顺序依次触发。向后台API请求的操作就在这个模块中进行,包括触发其他action以及提交mutation的操作。该模块提供了Promise的封装,以支持action的链式触发。
  • commit:状态改变提交操作方法。对mutation进行提交,是唯一能执行mutation的方法。
  • mutations:状态改变操作方法,由actions中的commit('mutation 名称')来触发。是Vuex修改state的唯一推荐方法。该方法只能进行同步操作,且方法名只能全局唯一。操作之中会有一些hook暴露出来,以进行state的监控等。
  • state:页面状态管理容器对象。集中存储Vue components中data对象的零散数据,全局唯一,以进行统一的状态管理。页面显示所需的数据从该对象中进行读取,利用Vue的细粒度数据响应机制来进行高效的状态更新。
  • getters:state对象读取方法。图中没有单独列出该模块,应该被包含在了render中,Vue Components通过该方法读取全局state对象。

Vuex与localStorage

vuex 是 vue 的状态管理器,存储的数据是响应式的。但是并不会保存起来,刷新之后就回到了初始状态,具体做法应该在vuex里数据改变的时候把数据拷贝一份保存到localStorage里面,刷新之后,如果localStorage里有保存的数据,取出来再替换store里的state。

六、跨级只传值 $attrs/$listeners

简介

多级组件嵌套需要传递数据时,通常使用的方法是通过vuex。但如果仅仅是传递数据,而不做中间处理,使用 vuex 处理,未免有点大材小用。为此Vue2.4 版本提供了另一种方法----$attrs/$listeners

  • $attrs:包含了父作用域中不被 prop 所识别 (且获取) 的特性绑定 (class 和 style 除外)。当一个组件没有声明任何 prop 时,这里会包含所有父作用域的绑定 (class 和 style 除外),并且可以通过 v-bind="$attrs" 传入内部组件。通常配合 interitAttrs 选项一起使用。
  • $listeners:包含了父作用域中的 (不含 .native 修饰器的) v-on 事件监听器。它可以通过 v-on="$listeners" 传入内部组件

七、$eventBus

$eventBus

$eventBus 主要作用是进行非父子组件信息传递

$eventBus

新建 event-bus.js 文件

// event-bus.js
import Vue from 'vue'
export const eventBus = new Vue()

第一步:在main.js中将$eventBus绑定到vue原型上

Vue.prototype.$eventBus = new Vue()

第二步:在需要传递信息的组件上将一个方法发射出去,并需要在某个条件激活这个方法

<div @click="busclick">按钮</div>
import { eventBus } from "../event-bus.js";

methods(){
  busclick(){
    this.$eventBus.$emit('busfunction')
  }
}

第三步:在组件的创建时created中接收emit的方法,并执行回调函数

created(){
  this.$eventBus.$on('busfunction',()=>{
    this.$refs.scroll.refresh()
  })
}

第四步:在beforeDestroy里销毁事件总线,不销毁的话会一直叠加的调用这个方法,如果这个子组件在不同的页面都调用这个子组件,那么在一个父组件this.off("busfunction");会把其他附件也销毁,如果只单独销毁一个父组件,那么需要在后面指定调用函数this.off("busfunction",callback());

beforDestroy(){
  this.$eventBus.$off("busfunction");  //当这个组件销毁的时候bus也跟着一起销毁
},
beforDestroy(){
  this.$eventBus.$off("busfunction",callback());  //当这个组件销毁的时候bus也跟着一起销毁
}

Vue3.X 使用eventbus

安装mitt依赖包

npm install mitt@2.1.0

创建eventBus.js文件

// 导入mitt包
import mitt from 'mitt'
// 创建eventbus的实例对象
const bus = mitt()

// 将eventbus的实例对象共享出去
export default bus

在数据接收方声明事件

在数据接收方,调用bus.on('事件名称', 事件处理函数)方法注册一个自定义事件。

import bus from './eventBus.js'
export default {
  data () {
    return {
      count: 0
    }
  },
  created () {
    bus.on('countChange', (count) => {
      this.count = count
    })
  }
}

在数据发送方触发事件

在数据发送方,调用bus.emit('事件名称', 要发送的数据)方法触发自定义事件。

import bus from './eventBus.js'
export default {
  data () {
    return {
      count: 0
    }
  },
  methods: {
    addCount () {
      this.count++
      bus.emit('countChange', this.count)
    }
  }
}