组件通信方式整合

83 阅读2分钟

image.png 在前端开发的工作中,组件通信是一个绕不开的技术需要。尤其是在开发复杂应用程序的过程中,为了实现后续的复用,我们往往需要通过封装组件来避免大量的代码重复.今天我想把平时工作中常用到的几种通信方法进行一个基本的整合,用来加深记忆.

一、基本概念

组件通信主要分为三个大的种类:父子组件间通信兄弟组件间通信以及任意组件之间

1、父子组件间通信

  • 子组件通过props属性来接收父组件的数据,然后父组件在子组件上注册监听事件,子组件通过$emit触发事件来向父组件发送数据。
  • 通过ref属性给子组件设置一个名字。父组件通过refs组件名来获得子组件,子组件通过refs组件名来获得子组件,子组件通过parent获得父子间,通过这种方式也可以实现通信。
  • 使用provide/inject,在父组件中通过provide提供变量,在子组件中通过inject来将变量注入到组件中。不论子组件有多深,只要调用了inject那么就注入provide中的数据.

2、兄弟组件间通信

  • 使用 eventBus 的方法,它的本质是通过创建一个空的 Vue 实例来作为消息传递的对象,通信的组件引入这个实例,通信的组件通过在这个实例上监听和触发事件,来实现消息的传递。
  • 通过 parent/parent/refs 来获取到兄弟组件,也可以进行通信。

3、任意组件之间

  • 使用 eventBus ,其实就是创建一个事件中心,相当于中转站,可以用它来传递事件和接收事件。

如果业务逻辑复杂,很多组件之间需要同时处理一些公共的数据,这个时候采用上面这一些方法可能不利于项目的维护。这个时候可以使用 vuex ,vuex 的思想就是将这一些公共的数据抽离出来,将它作为一个全局的变量来管理,然后其他组件就可以对这个公共数据进行读写操作,这样达到了解耦的目的。

二、通信方式的具体使用

方法一: props / $emit

1. 父组件向子组件传值

  • props只能是父组件向子组件进行传值,props使得父子组件之间形成了一个单向下行绑定。子组件的数据会随着父组件不断更新。
  • props 可以显示定义一个或一个以上的数据,对于接收的数据,可以是各种数据类型,同样也可以传递一个函数。
  • props属性名规则:若在props中使用驼峰形式,模板中需要使用短横线的形式

1687179462014.jpg

1687179575604.jpg

2. 子组件向父组件传值

  • $emit绑定一个自定义事件,当这个事件被执行的时就会将参数传递给父组件,而父组件通过v-on监听并接收参数。

d72ae5b1d34fa69433c90e31a6e2ac5.png 1687569174006.jpg

方法二:eventBus事件总线(emit/emit / on)

eventBus事件总线适用于父子组件非父子组件等之间的通信,事件总线就相当于一个桥梁,不同组件通过它来通信。使用步骤如下:

(1)创建事件中心管理组件之间的通信

1687569566230.jpg

(2)发送事件

假设有两个兄弟组件firstComsecondCom

1687569644822.jpg

firstCom组件中发送事件: 1687569690020.jpg

(3)接收事件

secondCom组件中发送事件:

image.png

方法三:依赖注入(provide / inject)

该方法用于父子组件之间的通信。当然这里所说的父子不一定是真正的父子,也可以是祖孙组件,在层数很深的情况下,可以使用这种方法来进行传值。就不用一层一层的传递了。依赖注入所提供的属性是非响应式的。 provide / inject是Vue提供的两个钩子,和datamethods是同级的。并且provide的书写形式和data一样。

  • provide 钩子用来发送数据或方法
  • inject钩子用来接收数据或方法

父组件中发送数据或方法

1687570010665.jpg

子组件中接收数据或方法

1687570130672.jpg

方法四:ref / $refs

实现父子组件之间的通信,ref这个属性用在子组件上,它的引用就指向了子组件的实例。可以通过实例来访问组件的数据和方法。

子组件中发送数据或方法

image.png

父组件中接收数据或方法 image.png

方法五:parent/parent / children

  • 使用$parent可以让组件访问父组件的实例(访问的是上一级父组件的属性和方法)
  • 使用$children可以让组件访问子组件的实例,但是,$children并不能保证顺序,并且访问的数据也不是响应式的。

1687570587370.jpg

1687570680581.jpg

在上面的代码中,子组件获取到了父组件的parentVal值,父组件改变了子组件中message的值。

需要注意:

  • 通过$parent访问到的是上一级父组件的实例,可以使用$root来访问根组件的实例
  • 在组件中使用$children拿到的是所有的子组件的实例,它是一个数组,并且是无序的
  • 在根组件#app上拿$parent得到的是new Vue()的实例,在这实例上再拿$parent得到的是undefined,而在最底层的子组件拿$children是个空数组
  • $children 的值是数组,而$parent是个对象