昨天使用了vue提供的事件发布订阅,于是今天就自己实现了一个,发现没有太大难度,直接上代码:
代码语言:javascript复制function EventBus() {}
EventBus.prototype.on = function (name, callback) {
//如果没有事件对象,新增一个
if(!this._events){
//创建一个干净的没有原型链的对象
this._events = Object.create(null);
}
//如果没有这个事件的订阅,新增一个,如果有,push进去
if(!this._events[name]){
this._events[name] = [callback];
}else{
this._events[name].push(callback);
}
}
EventBus.prototype.emit = function (name, ...args) {
//发布的时候,如果有这个事件,循环执行所有这个订阅的方法
if(this._events[name]){
this._events[name].forEach(callback => {
callback(...args);
})
}
}
EventBus.prototype.off = function (name) {
//如果有这个事件的订阅,清除所有订阅
if(this._events[name]){
delete this._events[name];
}
}
EventBus.prototype.once = function (name, callback) {
let once = (...args) => {
callback(...args);
this.off(name);
};
this.on(name, once);
}
let eventBus = new EventBus();
eventBus.on('on', function (msg) {
console.log(msg);
})
eventBus.once('once', function (msg) {
console.log(msg);
})
eventBus.on('off', function (msg) {
console.log(msg);
})
eventBus.emit('on', '发布on1')//发布on1
eventBus.emit('on', '发布on2')//发布on2
eventBus.emit('once', '发布once')//发布once
eventBus.emit('once', '发布once')
eventBus.emit('off', '发布off')//发布off
eventBus.off('off')
eventBus.emit('off', '发布off')
```
on、off和emit没什么好说的,就是很简单的发布订阅,代码看一下也都理解了,once这个方法我觉得得理解一下,once方法其实就是执行on,执行一次之后执行off,我们可以采用函数劫持的方法,把订阅的方法劫持了,依然是执行on,但是回调变成自己的,当自己的回调执行之后,off移除这个订阅,这种做法好像也叫切片编程。
代码语言:javascript复制EventBus.prototype.once = function (name, callback) {
let once = (...args) => {
callback(...args);
this.off(name);
};
this.on(name, once);
}
```
昨天有人问我eventBus具体应用场景,说实在的,我还真想不出来,兄弟组件之间传值、多层级组件之间传值。比如聊天点击进入一个页面,未读消息这种,大部分是在同一个页面展示,然后不同一个组件的场景才用得到,或者不同页面销毁一些会造成内存泄漏的场景。eventBus能解决的都有其它方法可以解决,其它方法也可以用eventBus解决,或许以后用到了eventBus之后,会觉得很香吧。
(完)