当前位置:首页 > 技术分析 > 正文内容

如何优雅地使用JS自定义一个全局事件总线(eventBus)?

ruisui883个月前 (02-03)技术分析22

我们的项目是基于Vue3的,众所周知,Vue3中移除了全局事件总线相关的API,那么,如果我们需要的话,该如何手动实现一个呢?今天,我教大家手写一个eventBus。

首先,我们先用文档注释定义一个类型 Fn,编写良好的文档注释,可以使我们像写TypeScript一样写JavaScript。

/** @typedef {Function & { once: boolean }} Fn */

事件总线通常至少需要4个API方法,及一个对象用于存储事件名称和回调函数。首先,我们先定义存储事件和回调的对象events

export default {
    /** @type {Object

接下来,我们创建emit方法,用于发射事件。该方法可接收无限个参数,第一个参数为事件名,后面的所有参数为事件携带的数据。我们使用es6展开运算符将剩余参数以args数组接收。我们根据事件名获取到该事件名对应的回调函数数组,如果存在该数组,那么遍历它,调用所有的回调函数,如果该函数的once属性的值为true,那么我们取消对该事件的监听。

export default {
  	... ...,
    emit (evt, ...args) {
      const fns = this.events[evt]
      fns && fns.forEach(fn => {
        fn(...args)
        fn.once && this.off(evt, fn)
      })
    }
}

然后,我们创建on方法,用于监听事件。该方法接收2个参数,第一个参数为事件名,第二个参数为回调函数。该方法非常简单,就是将fn添加进eventsevt对应的回调函数数组中。

export default {
  	... ...,
    on (evt, fn) {
      this.events[evt] = [...(this.events[evt] || []), fn]
    }
}

现在,我们创建once方法,用于仅监听一次事件。该方法与on方法接收同样的参数,唯一的不同是,我们首先将fnonce属性设置为true,用于标识该回调函数仅被触发一次,然后,调用on方法。

export default {
  	... ...,
    once (evt, /** @type {Fn} */ fn) {
      fn.once = true
      this.on(evt, fn)
    }
}

最后,我们创建off方法,用于注销事件监听,该方法与on接收同样的参数,我们首先获取到eventsevt事件名对应的回调函数数组fns,如果数组存在,我们获取fnfns中的索引,如果存在,则通过移除该回调函数。

export default {
  	... ...,
    off (evt, fn) {
      const fns = this.events[evt]
      if (fns) {
        const index = fns.indexOf(fn)
        index > -1 && fns.splice(index, 1)
      }
    }
}

感谢阅读!到此已经讲解完毕,自定义全局事件总线是不是很简单?大家有什么不明之处,或有更好的实现方案,欢迎评论区见!

后续有时间的话,我会分享更多Web开发相关的技术文章,欢迎大家阅读!

扫描二维码推送至手机访问。

版权声明:本文由ruisui88发布,如需转载请注明出处。

本文链接:http://www.ruisui88.com/post/717.html

分享给朋友:

“如何优雅地使用JS自定义一个全局事件总线(eventBus)?” 的相关文章

车辆费用统计管理系统

车辆费用统计管理系统headerfooter《车辆费用统计管理系统》是一款适用于中小型货运车辆管理公司统计车辆费用。主要包括车辆信息,费用管理,汇总统计等功能。本管理系统多处具有快速辅助录入功能,操作简单,易学、易用;处理功能高效强大,是协助您的好帮手!主要功能:1.基础设置:车辆档案、驾驶员信息2...

手把手教你Vue之父子组件间通信实践讲解【props、$ref 、$emit】

组件是 vue.js 最强大的功能之一,而组件实例的作用域是相互独立的,这就意味着不同组件之间的数据无法相互引用。那么组件间如何通信,也就成为了vue中重点知识了。这篇文章将会通过props、$ref和 $emit 这几个知识点,来讲解如何实现父子组件间通信。转载链接:https://www.jia...

GitLab-合并请求

描述合并请求可用于在您对项目进行的其他人员之间交换代码,并轻松与他们讨论更改。合并请求的步骤步骤1-在创建新的合并请求之前,GitLab中应该有一个创建的分支。您可以参考本章来创建分支-步骤2-登录到您的GitLab帐户,然后转到“ 项目”部分下的项目 -步骤3-单击“ 合并请求”选项卡,然后单击“...

前后端分离自动化运维平台开发

运维平台采用前后端分离:前端vue,框架vue-element-admin;后端python,框架django-rest-framework.目前运维平台模块如下:1、 CMDB管理应用管理、环境管理、开发语言管理、产品项目管理、资产管理2、 构建发布持续构建、持续部署、Jar工程依赖构建3、 容器...

HTML5学习笔记三:HTML5语法规则

1.标签要小写2.属性值可加可不加””或”3.可以省略某些标签 html body head tbody4.可以省略某些结束标签 tr td li例:显示效果:5.单标签不用加结束标签img input6.废除的标签font center big7.新添加的标签将在下一HTML5学习笔记中重点阐述。...

Acustica Audio 发布模拟Roland Jupiter 双声道合成器插件 TH2

福利: Acustica Audio 发布模拟Roland Jupiter 风格的双声道合成器插件 TH2 免费下载 意大利 Acustica Audio 公司发布布模拟Roland Jupiter 风格的双声道合成器插件 TH2 ,灵感来源于Acustica Audio的THING-8系列,它是...