Vue组件通信原理剖析(一)事件总线的基石 $on和$emit

首先我们先从一个面试题入手。

面试官问: “Vue中组件通信的常用方式有哪些?”
我答:
1. props
2. 自定义事件
3. eventbus
4. vuex
5. 还有常见的边界情况$parent、$children、$root、$refs、provide/inject
6. 此外还有一些非props特性$attrs、$listeners

面试官追问:“那你能分别说说他们的原理吗?”
我:[一脸懵逼]😳

下面我们就来一一看看他们内部的奥秘!
如果要看别的属性原理请移步到Vue组件通信原理剖析(二)全局状态管理Vuex和Vue组件通信原理剖析(三)provide/inject原理分析

props

解决问题:父给子传值

// child
props: {msg: {type: String,default: ''}
}// parent
<child msg="这是传给子组件的参数"></child>

自定义事件

解决问题:子给父传值

// child
this.$emit('add', '这是子组件传给父组件的参数')// parent
// parantAdd是定义在父组件中的事件,事件接受的参数$event就是子组件传给父组件的值
<child @add="parentAdd($event)"></child>

事件总线eventbus

解决问题:任意两个组件之间的传值

// 通常我们的做法是这样的
// main.js 
Vue.prototype.$bus = new Vue()// child1
this.$bus.$on('foo', handle)// child2
this.$bus.$meit('foo')

那么组件之间的通信到底是怎么实现的呢?on和on和onemit具体是怎么实现的?我们去源码中找一找答案,let’s go!

// $on 的实现逻辑
Vue.prototype.$on = function (event: string | Array<string>, fn: Function): Component {const vm: Component = thisif (Array.isArray(event)) {for (let i = 0, l = event.length; i < l; i++) {vm.$on(event[i], fn)}} else {(vm._events[event] || (vm._events[event] = [])).push(fn)}return vm}// $emit 的实现逻辑
Vue.prototype.$emit = function (event: string): Component {const vm: Component = thislet cbs = vm._events[event]if (cbs) {cbs = cbs.length > 1 ? toArray(cbs) : cbsconst args = toArray(arguments, 1)const info = `event handler for "${event}"`for (let i = 0, l = cbs.length; i < l; i++) {invokeWithErrorHandling(cbs[i], vm, args, vm, info)}}return vm}// invokeWithErrorHandling 的实现逻辑
export function invokeWithErrorHandling (handler: Function,context: any,args: null | any[],vm: any,info: string
) {let restry {res = args ? handler.apply(context, args) : handler.call(context)} catch (e) {handleError(e, vm, info)}return res
}

上面就是我们在源码中找到的实现,其中有一些调试代码我已经删除掉,方便大家可以抓住重点!
下面我们来一一分析

  1. 首先我们都了解vue的数据相应是依赖于“观察-订阅”模式,那on、on、onemit也不例外;
  2. $on用来收集所有的事件依赖,他会将传入的参数eventfn作为key和value的形式存到vm._events这个事件集合里,就像这样vm._events[event]=[fn];
  3. 而$emit是用来触发事件的,他会根据传入的eventvm_events中找到对应的事件并执行invokeWithErrorHandling(cbs[i], vm, args, vm, info)
  4. 最后我们看invokeWithErrorHandling方法可以发现,他是通过handler.apply(context, args)handler.call(context)的形式执行对应的方法

是不是很简单![偷笑]

我们既然知道怎么实现的,那么我们就可以自定义实现一个Bus, 看代码

// Bus: 事件派发、监听和回调
class Bus {constructor() {this.callbacks = {}}// 收集监听的回调函数$on(name, fn) {this.callbacks[name] = this.callbacks[name] || []this.callbacks[name].push(fn)}// 执行监听的回调函数$emit(name, args) {if (this.callbacks[name]) {this.callbacks[name].forEach(cb => cb(args))}}
}// 在main.js中这样使用
Vue.prototype.$bus = new Bus()

至此,关于总线的原理剖析就到这里。

全部文章链接

Vue组件通信原理剖析(一)事件总线的基石 on和on和onemit
Vue组件通信原理剖析(二)全局状态管理Vuex
Vue组件通信原理剖析(三)provide/inject原理分析

最后喜欢我的小伙伴也可以通过关注公众号“剑指大前端”,或者扫描下方二维码联系到我,进行经验交流和分享,同时我也会定期分享一些大前端干货,让我们的开发从此不迷路。
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/248809.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

display:flex弹性布局

一、背景 前段时间帮公司运维小姑娘调整她自己写的页面样式时发现她用了display: flex&#xff0c;我这个后端老古董还不太懂flex&#xff0c;自愧不如啊&#xff0c;所以写篇博客记录学习下。 现在写的前端页面还停留在依赖 display 属性 position属性 float属性的布局方式&…

一些好的思维方式

定理s 一、墨菲定律 观点&#xff1a;1.任何事都没有表面看起来那么简单&#xff1b;2.所有的事都会比你预计的时间长&#xff1b;3.会出错的事总会出错&#xff1b;4.如果你担心某种情况发生&#xff0c;那么它就更有可能发生。 墨菲定律的核心观点就4点&#xff0c;不算复杂&…

Vue组件通信原理剖析(二)全局状态管理Vuex

首先我们先从一个面试题入手。 面试官问&#xff1a; “Vue中组件通信的常用方式有哪些&#xff1f;” 我答&#xff1a; 1. props 2. 自定义事件 3. eventbus 4. vuex 5. 还有常见的边界情况$parent、$children、$root、$refs、provide/inject 6. 此外还有一些非props特性$att…

初识单点登录及JWT实现

单点登录 多系统&#xff0c;单一位置登录&#xff0c;实现多系统同时登录的一种技术 &#xff08;三方登录&#xff1a;某系统使用其他系统的用户&#xff0c;实现本系统登录的方式。如微信登录、支付宝登录&#xff09; 单点登录一般是用于互相授信的系统&#xff0c;实现单一…

Vue组件通信原理剖析(三)provide/inject原理分析

首先我们先从一个面试题入手。 面试官问&#xff1a; “Vue中组件通信的常用方式有哪些&#xff1f;” 我答&#xff1a; 1. props 2. 自定义事件 3. eventbus 4. vuex 5. 还有常见的边界情况$parent、$children、$root、$refs、provide/inject 6. 此外还有一些非props特性$att…

iMX6开发板-uboot-网络设置和测试

本文章基于迅为IMX6开发板 将iMX6开发板通过网线连接到路由器&#xff0c;同时连接好调试串口&#xff0c;上电立即按 enter&#xff0c;即可进入 uboot。然后输入命令 pri&#xff0c;查看开发板当前的配置&#xff0c;如下图所示可以看到 ip 地址、子网掩码 等信息。 本文档测…

Django ajax 检测用户名是否已被注册

添加一个 register.html 页面 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><title>Title</title> </head> <body> <form><p>用户名<input id"username" type&…

pyqt5控件

背景色设置 self.tab.setStyleSheet("background: rgb(238, 233, 233)") self.but_0.setStyleSheet("background: rgb(0, 255, 255)")样式&#xff1a; self.but_0.setStyle(QStyleFactory.create("Windows"))字体&#xff1a; self.lineEdit.se…

详解JDBC连接数据库

一、概念 1. 为了能让程序操作数据库&#xff0c;对数据库中的表进行操作&#xff0c;每一种数据库都会提供一套连接和操作该数据库的驱动&#xff0c;而且每种数据库的驱动都各不相同&#xff0c;例如mysql数据库使用mysql驱动&#xff0c;oracle数据库使用oracle驱动&#xf…

ASP.NET MVC 自定义模型绑定1 - 自动把以英文逗号分隔的 ID 字符串绑定成 Listint...

直接贴代码了&#xff1a; CommaSeparatedModelBinder.cs using System; using System.Collections; using System.Collections.Generic; using System.Linq; using System.Reflection; using System.Web.Mvc;namespace MvcSample.Extensions {public class CommaSeparatedMode…

ZOJ4024 Peak

题意 给出一个数组 判断这个数组是否形成了一个“山峰” 即中间有个数最大 从第一个数到这个数递增 从这个数到最后一个数递减 模拟 从两端分别以递增和递减判断 看第一个不满足递增或递减的数是否相等并且没越界就可以了 AC代码&#xff1a; 1 #include<bits/stdc.h>2 u…

基本数据类型与String之间的转换

字符串转基本数据类型 调用基本数据类型对应的包装类中的方法parseXXX(String)或valueOf(String)即可返回相应基本类型。 基本数据类型转字符串 一种方法是将基本数据类型与空字符串&#xff08;""&#xff09;连接&#xff08;&#xff09;即可获得其所对应的字符串…

springmvc跨域问题

1、跨域问题&#xff1a; 按照网上所有的方法试了一遍&#xff0c;都没跨过去&#xff0c;正在无助之际&#xff0c;使用filter按照下面的方法解决的时候出现了转机&#xff1a; 添加filter&#xff1a; package com.thc.bpm.filter;import javax.servlet.*; import javax.serv…

柳传志给年轻人的建议:比起过日子,更要奔日子

改革开放的 40 年&#xff0c;是柳传志实现人生价值的 40 年。 十一届三中全会后&#xff0c;伴随“科学的春天”&#xff0c;迎着改革开放的大潮&#xff0c;柳传志“下海”了。但他并没想到&#xff0c;自己选择的电脑行业&#xff0c;让他和联想集团站在了潮头。 从 1984 年…

成功秀了一波scala spark ML逻辑斯蒂回归

1、直接上官方代码&#xff0c;调整过的&#xff0c;方可使用 package com.test import org.apache.spark.{SparkConf, SparkContext} import org.apache.spark.mllib.classification.{LogisticRegressionModel, LogisticRegressionWithLBFGS} import org.apache.spark.mllib.e…

记录一次查询log的经历

一大早发现生产数据库的基础资料被删除。 由于每天都做了差异备份&#xff0c;而且是基础资料&#xff0c;这样数据就不会担心找不回来。 首先通过每天的差异本分文件进行查看数据丢失的大概时间&#xff0c;查到数据丢失是在17晚上备份过后18丢失的。 然后找18号的数据库执行记…

移动端轮播图

1. 页面布局 1.1 页面框架 <body><div class"box"><div class"tupian"><img src"4.webp" alt""><img src"1.webp" alt""><img src"2.webp" alt""><…

Boost 序列化

原文链接&#xff1a; https://blog.csdn.net/qq2399431200/article/details/45621921 1. 编译器 gcc, boost 1.55 2.1第一个简单的例子 —— Hello World &#xff0c;将字符串内容归档到文本文件中 #include <iostream>#include <fstream>#include <string>…

docker CE 的安装

一、Docker CE的安装1.先决条件运行环境&#xff1a;Ubuntu 64位或者其他支持Docker的64位系统运行配置&#xff0c;linux内核版本必须大于 3.10&#xff0c;否则会因为缺少容器运行所需的功能而出错。 2.在ubuntu下安装Docker CEUbuntu版本 Cosmic 18.10  Bionic 18.04 (…

nodeJS中的异步编程

nodejs 不是单线程 在博客项目中关于异步问题&#xff1a; 1.当用户添加一条博客时 需要通过post方式向服务器发送数据 后台获取用户以post方式拿到传送过来的数据 然后存入数据库&#xff1a; 上面的代码&#xff1a;创建一个空字符串 当用户向服务器发送请求时出发data事件将…