百度360必应搜狗淘宝本站头条
当前位置:网站首页 > IT技术 > 正文

vue 组件通信看这篇就够了(12种通信方式)

wptr33 2024-11-14 19:21 21 浏览


vue 组件间的通信是 vue 开发中很基础也十分重要的部分,作为使用 vue 的开发者每天都在使用。同时,vue 通信也是面试中非常高频的问题,有很多面试题,都是围绕通信展开。

本文会介绍常见的通信方式,并分析每种方式的使用场景和注意点。

vue中提倡单向数据流,这是为了保证数据流向的简洁性,使程序更易于理解。但对于一些边界情况,vue也提供了隐性的通信方式,这些通信方式会打破单向数据流的原则,应该谨慎使用。

下面我们将组件通信分为父子组件通信 和 非父子组件通信进行分析。

父子组件通信

prop 和 events

prop 和 events 最基础也最常用,这里不提供示例。

通过 prop 向下传递,通过事件向上传递是一个 vue 项目最理想的通信状态。

使用时有两点需要注意:

第一,不应该在一个子组件内部改变 prop,这样会破坏单向的数据绑定,导致数据流难以理解。如果有这样的需要,可以通过 data 属性接收或使用 computed 属性进行转换。

第二,如果 props 传递的是引用类型(对象或者数组),在子组件中改变这个对象或数组,父组件的状态会也会做相应的更新,利用这一点就能够实现父子组件数据的“双向绑定”,虽然这样实现能够节省代码,但会牺牲数据流向的简洁性,令人难以理解,最好不要这样去做。想要实现父子组件的数据“双向绑定”,可以使用 v-model 或 .sync。

v-model 指令

v-model 是用来在表单控件或者组件上创建双向绑定的,他的本质是 v-bind 和 v-on 的语法糖,在一个组件上使用 v-model,默认会为组件绑定名为 value 的 prop 和名为 input 的事件。

当我们组件中的某一个 prop 需要实现上面所说的”双向绑定“时,v-model 就能大显身手了。有了它,就不需要自己手动在组件上绑定监听当前实例上的自定义事件,会使代码更简洁。

下面以一个 input 组件实现的核心代码,介绍下 v-model 的应用。


<!--父组件-->
<template>
 <base-input v-model="input"></base-input>
</template>
<script>
 export default {
 data() {
 return {
 input: ''
 }
 },
 }
</script>

<!--子组件-->
<template>
 <input type="text" :value="currentValue" @input="handleInput">
</template>
<script>
 export default {
 data() {
 return {
 currentValue: this.value === undefined || this.value === null ? ''
 }
 },
 props: {
 value: [String, Number],
 },
 methods: {
 handleInput(event) {
 const value = event.target.value;
 this.$emit('input', value);
 },
 },
}
</script>

有时,在某些特定的控件中名为 value 的属性会有特殊的含义,这时可以通过 model 选项来回避这种冲突。

.sync 修饰符

.sync 修饰符在 vue 1.x 的版本中就已经提供,1.x 版本中,当子组件改变了一个带有 .sync 的 prop 的值时,会将这个值同步到父组件中的值。这样使用起来十分方便,但问题也十分明显,这样破坏了单向数据流,当应用复杂时,debug 的成本会非常高。

关注我的头条号,分享更多的技术学习文章,我自己是一名从事了多年开发的web前端老程序员,目前辞职在做自己的web前端私人定制课程,今年年初我花了一个月整理了一份最适合2019年学习的web前端学习干货,各种框架都有整理,送给每一位前端小伙伴,想要获取的可以关注我的头条号并在后台私信我:前端,即可免费获取。

在vue 2.0中移除了 .sync。但是在实际的应用中,.sync 是有它的应用场景的,所以在 vue 2.3 版本中,又迎来了全新的 .sync。

新的 .sync 修饰符所实现的已经不再是真正的双向绑定,它的本质和 v-model 类似,只是一种缩写。


<text-document
 v-bind:title="doc.title"
 v-on:update:title="doc.title = $event"
></text-document>

上面的代码,使用 .sync 就可以写成


<text-document v-bind:title.sync="doc.title"></text-document>

这样,在子组件中,就可以通过下面代码来实现对这个 prop 重新赋值的意图了。


this.$emit('update:title', newTitle)

v-model 和 .sync 对比

.sync 从功能上看和 v-model 十分相似,都是为了实现数据的“双向绑定”,本质上,也都不是真正的双向绑定,而是语法糖。

相比较之下,.sync 更加灵活,它可以给多个 prop 使用,而 v-model 在一个组件中只能有一个。

从语义上来看,v-model 绑定的值是指这个组件的绑定值,比如 input 组件,select 组件,日期时间选择组件,颜色选择器组件,这些组件所绑定的值使用 v-model 比较合适。其他情况,没有这种语义,个人认为使用 .sync 更好。

ref

ref 特性可以为子组件赋予一个 ID 引用,通过这个 ID 引用可以直接访问这个子组件的实例。当父组件中需要主动获取子组件中的数据或者方法时,可以使用 $ref 来获取。


<!--父组件-->
<template>
 <base-input ref="baseInput"></base-input>
</template>
<script>
 export default {
 methods: {
 focusInput: function () {
 this.$refs.usernameInput.focus()
 }
 }
}
</script>

<!--子组件-->
<template>
 <input ref="input">
</template>
<script>
 export default {
 methods: {
 focus: function () {
 this.$refs.input.focus()
 }
 }
}
</script>

使用 ref 时,有两点需要注意

  • $refs 是作为渲染结果被创建的,所以在初始渲染的时候它还不存在,此时无法无法访问。
  • $refs 不是响应式的,只能拿到获取它的那一刻子组件实例的状态,所以要避免在模板和计算属性中使用它。

$parent 和 $children

$parent 属性可以用来从一个子组件访问父组件的实例,$children 属性 可以获取当前实例的直接子组件。

看起来使用 $parent 比使用prop传值更加简单灵活,可以随时获取父组件的数据或方法,又不像使用 prop 那样需要提前定义好。但使用 $parent 会导致父组件数据变更后,很难去定位这个变更是从哪里发起的,所以在绝大多数情况下,不推荐使用。

在有些场景下,两个组件之间可能是父子关系,也可能是更多层嵌套的祖孙关系,这时就可以使用 $parent。

下面是 element ui 中的组件 el-radio-group 和 组件 el-radio 使用示例:


<template>
 <el-radio-group v-model="radio1">
 <el-radio :label="3">备选项</el-radio>
 <component-1>
 <el-radio :label="3">备选项</el-radio>
 </component-1>
 </el-radio-group>
</template>
<script>
 export default {
 data () {
 return {
 radio2: 3
 };
 }
 }
</script>

在 el-radio-group 和 组件 el-radio 通信中, 组件 el-radio 的 value 值需要和 el-radio-group的 v-model 的值进行“绑定”,我们就可以在 el-radio 内借助 $parent 来访问到 el-radio-group 的实例,来获取到 el-radio-group 中 v-model 绑定的值。

下面是获取 el-radio 组件中获取 el-radio-group 实例的源码:


// el-radio组件
 let parent = this.$parent;
 while (parent) {
 if (parent.$options.componentName !== 'ElRadioGroup') {
 parent = parent.$parent;
 } else {
 this._radioGroup = parent; // this._radioGroup 为组件 el-radio-group 的实例
 }
 }

非父子组件通信

$attrs 和 $listeners

当要和一个嵌套很深的组件进行通信时,如果使用 prop 和 events 就会显的十分繁琐,中间的组件只起到了一个中转站的作用,像下面这样:


<!--父组件-->
 <parent-component :message="message">我是父组件</parent-component>
<!--子组件-->
 <child-component :message="message">我是子组件</child-component>
<!--孙子组件-->
 <grand-child-component :message="message">我是孙子组件</grand-child-component>

当要传递的数据很多时,就需要在中间的每个组件都重复写很多遍,反过来从后代组件向祖先组件使用 events 传递也会有同样的问题。使用 $attrs 和 $listeners 就可以简化这样的写法。

$attrs 会包含父组件中没有被 prop 接收的所有属性(不包含class 和 style 属性),可以通过 v-bind="$attrs" 直接将这些属性传入内部组件。

$listeners 会包含所有父组件中的 v-on 事件监听器 (不包含 .native 修饰器的) ,可以通过 v-on="$listeners" 传入内部组件。

下面以父组件和孙子组件的通信为例介绍它们的使用:


<!--父组件 parent.vue-->
<template>
 <child :name="name" :message="message" @sayHello="sayHello"></child>
</template>
<script>
export default {
 inheritAttrs: false,
 data() {
 return {
 name: '通信',
 message: 'Hi',
 }
 },
 methods: {
 sayHello(mes) {
 console.log('mes', mes) // => "hello"
 },
 },
}
</script>

<!--子组件 child.vue-->
<template>
 <grandchild v-bind="$attrs" v-on="$listeners"></grandchild>
</template>
<script>
export default {
 data() {
 return {}
 },
 props: {
 name,
 },
}
</script>

<!--孙子组件 grand-child.vue-->
<template>
</template>
<script>
export default {
 created() {
 this.$emit('sayHello', 'hello')
 },
}
</script>

provide 和 inject

provide 和 inject 需要在一起使用,它可以使一个祖先组件向其所有子孙后代注入一个依赖,可以指定想要提供给后代组件的数据/方法,不论组件层次有多深,都能够使用。


<!--祖先组件-->
<script>
export default {
 provide: {
 author: 'yushihu',
 },
 data() {},
}
</script>

<!--子孙组件-->
<script>
export default {
 inject: ['author'],
 created() {
 console.log('author', this.author) // => yushihu
 },
}
</script>

provide 和 inject 绑定不是响应的,它被设计是为组件库和高阶组件服务的,平常业务中的代码不建议使用。

dispatch 和 broadcast

vue 在2.0版本就已经移除了 $dispatch 和 $broadcast,因为这种基于组件树结构的事件流方式会在组件结构扩展的过程中会变得越来越难维护。但在某些不使用 vuex 的情况下,仍然有使用它们的场景。所以 element ui 和 iview 等开源组件库中对 broadcast 和 dispatch 方法进行了重写,并通过 mixin 的方式植入到每个组件中。

实现 dispatch 和 broadcast 主要利用我们上面已经说过的 $parent 和 $children。


//element ui 中重写 broadcast 的源码
function broadcast(componentName, eventName, params) {
 this.$children.forEach(child => {
 var name = child.$options.componentName;
 if (name === componentName) {
 child.$emit.apply(child, [eventName].concat(params));
 } else {
 broadcast.apply(child, [componentName, eventName].concat([params]));
 }
 });
}

broadcast 方法的作用是向后代组件传值,它会遍历所有的后代组件,如果后代组件的 componentName 与当前的组件名一致,则触发 $emit 事件,将数据 params 传给它。


 //element ui 中重写 dispatch 的源码
 dispatch(componentName, eventName, params) {
 var parent = this.$parent || this.$root;
 var name = parent.$options.componentName;
 while (parent && (!name || name !== componentName)) {
 parent = parent.$parent;
 if (parent) {
 name = parent.$options.componentName;
 }
 }
 if (parent) {
 parent.$emit.apply(parent, [eventName].concat(params));
 }
 }

dispatch 的作用是向祖先组件传值,它会一直寻找父组件,直到找到组件名和当前传入的组件名一致的祖先组件,就会触发其身上的 $emit 事件,将数据传给它。这个寻找对应的父组件的过程和文章前面讲解 $parent 的例子类似。

eventBus

对于比较小型的项目,没有必要引入 vuex 的情况下,可以使用 eventBus。相比我们上面说的所有通信方式,eventBus 可以实现任意两个组件间的通信。

它的实现思想也很好理解,在要相互通信的两个组件中,都引入同一个新的vue实例,然后在两个组件中通过分别调用这个实例的事件触发和监听来实现通信。


//eventBus.js
import Vue from 'vue';
export default new Vue();

<!--组件A-->
<script>
import Bus from 'eventBus.js';
export default {
 methods: {
 sayHello() {
 Bus.$emit('sayHello', 'hello');
 }
 }
}
</script>

<!--组件B-->
<script>
import Bus from 'eventBus.js';
export default {
 created() {
 Bus.$on('sayHello', target => {
 console.log(target); // => 'hello'
 });
 }
}
</script>

通过 $root 访问根实例

通过 $root,任何组件都可以获取当前组件树的根 Vue 实例,通过维护根实例上的 data,就可以实现组件间的数据共享。


//main.js 根实例
new Vue({
 el: '#app',
 store,
 router,
 // 根实例的 data 属性,维护通用的数据
 data: function () {
 return {
 author: ''
 }
 },
 components: { App },
 template: '<App/>',
});

<!--组件A-->
<script>
export default {
 created() {
 this.$root.author = '于是乎'
 }
}
</script>

<!--组件B-->
<template>
 <div><span>本文作者</span>{{ $root.author }}</div>
</template>

通过这种方式,虽然可以实现通信,但在应用的任何部分,任何时间发生的任何数据变化,都不会留下变更的记录,这对于稍复杂的应用来说,调试是致命的,不建议在实际应用中使用。

Vuex

Vuex 是一个专为 Vue.js 应用程序开发的状态管理模式。它采用集中式存储管理应用的所有组件的状态,并以相应的规则保证状态以一种可预测的方式发生变化。对一个中大型单页应用来说是不二之选。

使用 Vuex 并不代表就要把所有的状态放入 Vuex 管理,这样做会让代码变的冗长,无法直观的看出要做什么。对于严格属于组件私有的状态还是应该在组件内部管理更好。

自己实现简单的 Store 模式

对于小型的项目,通信十分简单,这时使用 Vuex 反而会显得冗余和繁琐,这种情况最好不要使用 Vuex,可以自己在项目中实现简单的 Store。


//store.js
var store = {
 debug: true,
 state: {
 author: 'yushihu!'
 },
 setAuthorAction (newValue) {
 if (this.debug) console.log('setAuthorAction triggered with', newValue)
 this.state.author = newValue
 },
 deleteAuthorAction () {
 if (this.debug) console.log('deleteAuthorAction triggered')
 this.state.author = ''
 }
}

和 Vuex 一样,store 中 state 的改变都由 store 内部的 action 来触发,并且能够通过 log 保留触发的痕迹。这种方式十分适合在不需要使用 Vuex 的小项目中应用。

与 $root 访问根实例的方法相比,这种集中式状态管理的方式能够在调试过程中,通过 log 记录来确定当前变化是如何触发的,更容易定位问题。

相关推荐

Python自动化脚本应用与示例(python办公自动化脚本)

Python是编写自动化脚本的绝佳选择,因其语法简洁、库丰富且跨平台兼容性强。以下是Python自动化脚本的常见应用场景及示例,帮助你快速上手:一、常见自动化场景文件与目录操作...

Python文件操作常用库高级应用教程

本文是在前面《Python文件操作常用库使用教程》的基础上,进一步学习Python文件操作库的高级应用。一、高级文件系统监控1.1watchdog库-实时文件系统监控安装与基本使用:...

Python办公自动化系列篇之六:文件系统与操作系统任务

作为高效办公自动化领域的主流编程语言,Python凭借其优雅的语法结构、完善的技术生态及成熟的第三方工具库集合,已成为企业数字化转型过程中提升运营效率的理想选择。该语言在结构化数据处理、自动化文档生成...

14《Python 办公自动化教程》os 模块操作文件与文件夹

在日常工作中,我们经常会和文件、文件夹打交道,比如将服务器上指定目录下文件进行归档,或将爬虫爬取的数据根据时间创建对应的文件夹/文件,如果这些还依靠手动来进行操作,无疑是费时费力的,这时候Pyt...

python中os模块详解(python os.path模块)

os模块是Python标准库中的一个模块,它提供了与操作系统交互的方法。使用os模块可以方便地执行许多常见的系统任务,如文件和目录操作、进程管理、环境变量管理等。下面是os模块中一些常用的函数和方法:...

21-Python-文件操作(python文件的操作步骤)

在Python中,文件操作是非常重要的一部分,它允许我们读取、写入和修改文件。下面将详细讲解Python文件操作的各个方面,并给出相应的示例。1-打开文件...

轻松玩转Python文件操作:移动、删除

哈喽,大家好,我是木头左!Python文件操作基础在处理计算机文件时,经常需要执行如移动和删除等基本操作。Python提供了一些内置的库来帮助完成这些任务,其中最常用的就是os模块和shutil模块。...

Python 初学者练习:删除文件和文件夹

在本教程中,你将学习如何在Python中删除文件和文件夹。使用os.remove()函数删除文件...

引人遐想,用 Python 获取你想要的“某个人”摄像头照片

仅用来学习,希望给你们有提供到学习上的作用。1.安装库需要安装python3.5以上版本,在官网下载即可。然后安装库opencv-python,安装方式为打开终端输入命令行。...

Python如何使用临时文件和目录(python目录下文件)

在某些项目中,有时候会有大量的临时数据,比如各种日志,这时候我们要做数据分析,并把最后的结果储存起来,这些大量的临时数据如果常驻内存,将消耗大量内存资源,我们可以使用临时文件,存储这些临时数据。使用标...

Linux 下海量文件删除方法效率对比,最慢的竟然是 rm

Linux下海量文件删除方法效率对比,本次参赛选手一共6位,分别是:rm、find、findwithdelete、rsync、Python、Perl.首先建立50万个文件$testfor...

Python 开发工程师必会的 5 个系统命令操作库

当我们需要编写自动化脚本、部署工具、监控程序时,熟练操作系统命令几乎是必备技能。今天就来聊聊我在实际项目中高频使用的5个系统命令操作库,这些可都是能让你效率翻倍的"瑞士军刀"。一...

Python常用文件操作库使用详解(python文件操作选项)

Python生态系统提供了丰富的文件操作库,可以处理各种复杂的文件操作需求。本教程将介绍Python中最常用的文件操作库及其实际应用。一、标准库核心模块1.1os模块-操作系统接口主要功能...

11. 文件与IO操作(文件io和网络io)

本章深入探讨Go语言文件处理与IO操作的核心技术,结合高性能实践与安全规范,提供企业级解决方案。11.1文件读写11.1.1基础操作...

Python os模块的20个应用实例(python中 import os模块用法)

在Python中,...