js手写事件Event、手写订阅模式、手写event.js

简介: 观察者模式或者说订阅模式,它定义了对象间的一种一对多的关系,让多个观察者对象同时监听某一个主题对象,当一个对象发生改变时,所有依赖于它的对象都将得到通知。

通俗来说,就像订报纸一样,只要你订阅了这个报纸,当新报发布的时候你就会收到新报纸的推送。许多场景都运用了这种订阅模式,如vue.$onvue.$emitNode中的EventEmitter等。

其实现也是相对来说比较简单的,废话不多说,直接上代码

class Event {
    constructor() {
        this.handlers = {}
    }

    on(eventName, cb) {
        const eventCallbackStack = this._getHandler(eventName).callbackStack
        eventCallbackStack.push(cb)
    }

    emit(eventName, ...args) {
        if(this.handlers[eventName]) {
            this.handlers[eventName].callbackStack.forEach(cb => {
                // 修正this指向
                cb.call(cb, ...args)
            })
            // 移除once事件
            if(this.handlers[eventName].isOnce) {
                this.off(eventName)
            }
        }
    }

    off(eventName) {
        this.handlers[eventName] && delete this.handlers[eventName]
    }

    once(eventName, cb) {
        const eventCallbackStack = this._getHandler(eventName, true).callbackStack
        eventCallbackStack.push(cb)
    }

    /**
     * 根据事件名获取事件对象
     * @param eventName
     * @param isOnce  // 是否为once事件
     */
    _getHandler(eventName, isOnce = false){
        if(!this.handlers[eventName]) {
            this.handlers[eventName] = {
                isOnce,
                callbackStack: [],
            }
        }
        return this.handlers[eventName]
    }
}
  • 测试on事件
const event = new Event()
console.log('注册listen事件')
event.on('listen', function(a, b){
    console.log('监听到listen事件:', a, b)
})

console.log('触发listen事件')
event.emit('listen', 1, 2)
console.log('注销listen事件')
event.off('listen')
console.log('再次触发listen事件')
event.emit('listen', 1, 2)

在这里插入图片描述
可以看到on事件是可以off注销掉的,注销之后就没有再次触发了

  • 测试once事件
console.log('注册once事件')
event.once('once', function (a,b) {
    console.log('监听到once事件:',a, b)
})
console.log('触发once事件')
event.emit('once', 1, 2)
console.log('再次触发once事件')
event.emit('once', 1, 2)

在这里插入图片描述
可以看到once事件只触发一次

  • 2
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值