深入理解Vue $on、$emit、$off

缘起

遇到一个问题(vue2.x版本下):

A组件、D组件需要通信,考虑到嵌套的层级比较多,所以采用了eventBus的方式传递数据。

D组件因为某些操作触发了组件更新后,无法再成功监听到来自A组件传递的数据。

排查发现,问题出在了D组件移除事件的调用方式上,上述的写法中,只提供了事件,Vue会移除该事件所有的监听器

修改D组件监听及移除事件的方式,问题成功解决。

所以在这个问题的场景里,移除事件监听器时,需要提供回调函数,只移除这个回调的监听器,才能保证下一次更新组件时能正常监听到事件。

回顾

查阅Vue官方文档(Vue2.x),回顾下Vue提供的三个API的功能特性:

触发事件

vm.$emit( eventName, [...args] )

触发当前实例上的事件,eventName为事件名称,args为需要传递的参数、数据。

监听事件

vm.$on( event, callback )

监听当前实例上的自定义事件, event事件名需与触发事件时定义的事件名相同; callback为回调函数,在里面可以执行监听到事件后需要执行的操作。

移除事件

vm.$off( [event, callback] )

移除自定义事件监听器。这里提供了三种使用场景:

1.不提供参数,则移除所有的事件监听器;

bash 复制代码
vm.$off();

2.如果只提供了事件,则移除该事件所有的监听器;

bash 复制代码
vm.$off(eventName);

3.如果同时提供了事件与回调,则只移除这个回调的监听器。

bash 复制代码
vm.$off(eventName, callback);

深入

查看下Vue(这里基于2.7.14)源码中对上述三个API的实现方式:

$emit

  • 遍历回调函数列表
  • 调用每个回调函数并传递参数

$on

  • 如果传入的事件是数组,则遍历数组,并对每个子项调用$on
  • 如果传入的是单个事件:
    • 如果传入的事件不存在,创建一个新数组,并添加回调函数
    • 如果传入的事件已经存在,则将新传入的回调函数添加到事件数组列表中

$off

  • 如果传入的事件是数组,则遍历数组,并对每个子项调用$off
  • 判断是否传入了指定的回调函数,如果没传入,则移除所有监听器。若传入了,则移除指定回调的监听器

模拟实现事件总线

kotlin 复制代码
class Bus {
    constructor() {
        this.busEvent = {};
        this.$on = (name, func) => {
            if (this.busEvent[name] ) {
                this.busEvent[name][Symbol.for(func)] = func;
            } else {
                this.busEvent[name] = {}
                this.busEvent[name][Symbol.for(func)] = func;
            }
        }
        this.$off = (name, func) => {
            if (func) {
                this.busEvent[name][Symbol.for(func)] = undefined
            } else {
                this.busEvent[name] = undefined
            }
        }
        this.$emit = (name) => {
            if (!this.busEvent[name]) return
            Object.getOwnPropertySymbols(this.busEvent[name]).forEach(el => {
                console.log(this.busEvent[name][el], 'hihihi')
                if (this.busEvent[name][el] !== undefined) this.busEvent[name][el]()
            })
        }
    }
}

const bus = new Bus()
bus.$on('test', () => {
    console.log('func1111')
})
let func2 = () => {
    console.log('func222222')
}

bus.$emit('test')
bus.$on('test', func2)
bus.$off('test', func2)
// bus.$off('test', () => {
//     console.log('func222222')
// })

Vue3.x中的变更

在Vue3.x版本下,$on$off$once 实例方法已被移除,组件实例不再实现事件触发接口。$emit 仍然包含于现有的 API 中,因为它用于触发由父组件声明式添加的事件处理函数

这意味着在Vue3.x版本中,Vue不再支持使用事件总线的方式进行组件间的通信,需要借助第三方的库实现。例如 mitttiny-emitter

另外,作者也表示,在绝大多数情况下,不鼓励使用全局的事件总线在组件之间进行通信。

可以使用一些替代方案。例如provide / injectPinia

相关推荐
KaMeidebaby19 小时前
卡梅德生物技术快报|骆驼纳米抗体:从原核表达、高通量测序到分子对接全流程实现
前端·数据库·其他·百度·新浪微博
子兮曰21 小时前
Node.js v26.1.0 深度解读:FFI、后量子密码与调试器的进化
前端·后端·node.js
测试员周周1 天前
【Appium 系列】第06节-页面对象实现 — LoginPage 实战
开发语言·前端·人工智能·python·功能测试·appium·测试用例
西洼工作室1 天前
前端直传OSS服务端签名(Policy+Signature)/STS临时凭证
前端·文件上传·oss
你很易烊千玺1 天前
日常练习-数组 字符串常用的场景
前端·javascript·字符串·数组
weixin199701080161 天前
[特殊字符] RESTful API 接口规范详解:构建高效、可扩展的 Web 服务(附 Python 源码)
前端·python·restful
存在的五月雨1 天前
Vue3项目一些语法
前端·javascript·react.js
nashane1 天前
HarmonyOS 6学习:Web组件同层渲染事件处理与智能长截图实现
前端·学习·harmonyos·harmonyos 5
大家的林语冰1 天前
Node 2026 发布,JS 三大新功能上线,最后一个奇偶版本
前端·javascript·node.js