vuejs 传参 向 子组件 父组件_vue 组件的通信方式(完整版)

本文详细介绍了Vue.js中父子组件间的通信方式,包括Prop、$emit、Sync修饰符、v-bind和v-on的使用,以及props、events、provide/inject等,强调了单向数据流的重要性并提供了示例代码。
摘要由CSDN通过智能技术生成
几种通信方式无外乎以下几种:

Prop(常用)

$emit (组件封装用的较多)

.sync语法糖 (较少)

$attrs & $listeners (组件封装用的较多)

provide & inject (高阶组件/组件库用的较多)

slot-scope & v-slot ([email protected]+)新增

scopedSlots 属性

其他方式通信

下面逐个介绍,大神请绕行。

1. Prop

英式发音:[pr?p]。这个在我们日常开发当中用到的非常多。简单来说,我们可以通过 Prop 向子组件传递数据。用一个形象的比喻来说,父子组件之间的数据传递相当于自上而下的下水管子,只能从上往下流,不能逆流。这也正是 Vue 的设计理念之单向数据流。而 Prop 正是管道与管道之间的一个衔接口,这样水(数据)才能往下流。说这么多,看代码:

676243fdc2a75a3e09862cf97f684a6d.png

2. $emit

英式发音:[i?m?t]。官方说法是触发当前实例上的事件。附加参数都会传给监听器回调。按照我的理解不知道能不能给大家说明白,先简单看下代码吧:

cbc5025f3c7ce0dd79d619960c5d08c5.png

你可以狠狠的戳这里查看Demo! 大致逻辑是酱婶儿的:当我在页面上点击按钮时,触发了组件 MyButton 上的监听事件 greet,并且把参数传给了回调函数 sayHi 。说白了,当我们从子组件 Emit(派发) 一个事件之前,其内部都提前在事件队列中 On(监听)了这个事件及其监听回调。其实相当于下面这种写法:

afd5a42944b3971bc8bdd4fcb18a5d8d.png

3. Sync 修饰符

这个家伙在 [email protected] 的时候曾作为双向绑定功能存在,即子组件可以修改父组件中的值。因为它违反了单向数据流的设计理念,所以在 [email protected] 的时候被干掉了。但是在 [email protected]+ 以上版本又重新引入了这个 .sync 修饰符。但是这次它只是作为一个编译时的语法糖存在。它会被扩展为一个自动更新父组件属性的 v-on 监听器。说白了就是让我们手动进行更新父组件中的值了,从而使数据改动来源更加的明显。下面引入自官方的一段话:

在有些情况下,我们可能需要对一个 prop 进行“双向绑定”。不幸的是,真正的双向绑定会带来维护上的问题,因为子组件可以修改父组件,且在父组件和子组件都没有明显的改动来源。

既然作为一个语法糖,肯定是某种写法的简写形式,哪种写法呢,看代码:

ec84829484946585426c22f5ec364bd0.png

于是我们可以用 .sync 语法糖简写成如下形式:

5332e3cdd3ff7e4777ebd459d17ce55d.png

假如我们想实现这样一个效果:改变子组件文本框中的值同时改变父组件中的值。怎么做?

977fa0334f34163862b306eca3404df1.png

你可以狠狠的戳这里查看Demo!下面划重点,代码里有这一句话:

719323ab4713d250dd3ec42dd75c015b.png

官方语法是: update:myPropName  其中  myPropName  表示要更新的 prop 值。当然如果你不用 .sync 语法糖使用上面的 .$emit 也能达到同样的效果。仅此而已!

4. $attrs 和 $listeners

官网对 $attrs 的解释如下:

包含了父作用域中不作为 prop 被识别 (且获取) 的特性绑定 (class 和 style 除外)。当一个组件没有声明任何 prop 时,这里会包含所有父作用域的绑定 (class 和 style 除外),并且可以通过 v-bind="$attrs" 传入内部组件——在创建高级别的组件时非常有用。

官网对 $listeners 的解释如下:

包含了父作用域中的 (不含 .native 修饰器的) v-on 事件监听器。它可以通过 v-on="$listeners" 传入内部组件——在创建更高层次的组件时非常有用。

我觉得  $attrs  和  $listeners  属性像两个收纳箱,一个负责收纳属性,一个负责收纳事件,都是以对象的形式来保存数据。看下面的代码解释:

f7c491ffde619cf5c4998b74f1e3a419.png

从 Html 中可以看到,这里有俩属性和俩方法,区别是属性一个是 prop 声明,事件一个是 .native 修饰器。

af32ac51d4a53f2901cc0085b4a49a08.png

你可以狠狠的戳这里查看Demo! 可以看到,我们可以通过  $attrs  和  $listeners  进行数据传递,在需要的地方进行调用和处理,还是很方便的。当然,我们还可以通过  v-on="$listeners"  一级级的往下传递,子子孙孙无穷尽也!同时,子孙如果用了 $listeners 的方法,也可以通过 $emit 向该方法的源头传参,这个源头的方法就可以接收到该子孙传来的参数。 一个插曲! 当我们在组件上赋予了一个非Prop 声明时,编译之后的代码会把这些个属性都当成原始属性对待,添加到 html 原生标签上,看上面的代码编译之后的样子:

da05ff2758c27cb69d9e8e898e486522.png

这样会很难看,同时也爆了某些东西。如何去掉?这正是 inheritAttrs 属性的用武之地!给组件加上这个属性就行了,一般是配合 $attrs 使用。看代码:

fd4156166b2bd62a5b1a3d9ff597f44b.png

再次编译:

2cafa26ea0dd9b98c72fde654ed5de3b.png

5. provide / inject

他俩是对CP, 感觉挺神秘的。来看下官方对 provide / inject 的描述:

provide 和 inject 主要为高阶插件/组件库提供用例。并不推荐直接用于应用程序代码中。并且这对选项需要一起使用,以允许一个祖先组件向其所有子孙后代注入一个依赖,不论组件层次有多深,并在起上下游关系成立的时间里始终生效。

这对选项需要一起使用,以允许一个祖先组件向其所有子孙后代注入一个依赖,不论组件层次有多深,并在起上下游关系成立的时间里始终生效。一言而蔽之:祖先组件中通过provider来提供变量,然后在子孙组件中通过inject来注入变量。 provide / inject API 主要解决了跨级组件间的通信问题,不过它的使用场景,主要是子组件获取上级组件的状态,跨级组件间建立了一种主动提供与依赖注入的关系。 看完描述有点懵懵懂懂!一句话总结就是:小时候你老爸什么东西都先帮你存着等你长大该娶媳妇儿了你要房子给你买要车给你买只要他有的尽量都会满足你。 下面是这句话的代码解释:

3cf5ecd6895aa3c196f1d47ad8884536.png

你可以狠狠的戳这里查看Demo!

6. slot-scope & v-slot

关于这种方式的介绍,请看我这篇文章的介绍。传送门->

7. scopedSlots 属性

关于这种方式的介绍,请看我这篇文章的介绍。传送门->

8. 其他方式通信

除了以上五种方式外,其实还有: EventBus 思路就是声明一个全局Vue实例变量  EventBus  , 把所有的通信数据,事件监听都存储到这个变量上。可以阅读 使用Event Bus进行Vue组件间通信了解更多。这样就达到在组件间数据共享了,有点类似于 Vuex。但这种方式只适用于极小的项目,复杂项目还是推荐 Vuex。下面是实现 EventBus 的简单代码:

751635771c0bf46a20c19b0606ad8775.png

你可以狠狠的戳这里查看Demo! Vuex 官方推荐的,Vuex 是一个专为 Vue.js 应用程序开发的状态管理模式。 $parent 父实例,如果当前实例有的话。通过访问父实例也能进行数据之间的交互,但极小情况下会直接修改父组件中的数据。 $root 当前组件树的根 Vue 实例。如果当前实例没有父实例,此实例将会是其自己。通过访问根组件也能进行数据之间的交互,但极小情况下会直接修改父组件中的数据。 broadcast / dispatch 他俩是 [email protected] 中的方法,分别是事件广播 和 事件派发。虽然 [email protected] 里面删掉了,但可以模拟这两个方法。可以借鉴 Element 实现。有时候还是非常有用的,比如我们在开发树形组件的时候等等。 编辑整理丨刘学华

2f55a51d0900b8e4ce95d2dd9d5900c1.png

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值