深入理解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

相关推荐
yddddddy14 小时前
css的基本知识
前端·css
昔人'14 小时前
css `lh`单位
前端·css
Nan_Shu_61416 小时前
Web前端面试题(2)
前端
知识分享小能手16 小时前
React学习教程,从入门到精通,React 组件核心语法知识点详解(类组件体系)(19)
前端·javascript·vue.js·学习·react.js·react·anti-design-vue
蚂蚁RichLab前端团队17 小时前
🚀🚀🚀 RichLab - 花呗前端团队招贤纳士 - 【转岗/内推/社招】
前端·javascript·人工智能
孩子 你要相信光17 小时前
css之一个元素可以同时应用多个动画效果
前端·css
huangql52017 小时前
npm 发布流程——从创建组件到发布到 npm 仓库
前端·npm·node.js
Days205018 小时前
LeaferJS好用的 Canvas 引擎
前端·开源
小白菜学前端18 小时前
vue2 常用内置指令总结
前端·vue.js
林_深时见鹿18 小时前
Vue + ElementPlus 自定义指令控制输入框只可以输入数字
前端·javascript·vue.js