【常见面试题】JS 发布者、订阅者模式

面试中经常出现问到如何实现JS 发布者、订阅者模式。下面是ES5实现发布订阅模式。

1、直接上代码。

javascript 复制代码
function EventEmitter() {
    this.events = {};
};
// 订阅者
EventEmitter.prototype.on = function(ename, callback) {
    if (!this.events[ename]) {
        // 初始化创建订阅,一个订阅名可以创建多个时间函数
        this.events[ename] = [callback];
    } else {
        // 订阅已存在,push指定订阅名称尾部
        this.events[ename].push(callback);
    }
}
// 发布者
EventEmitter.prototype.emit = function(ename) {
    // 遍历执行订阅名称下的所有订阅者事件
    this.events[ename] && this.events[ename].forEach(cb => {
        // 执行订阅者函数
        cb();
    });
}
// 发布者
EventEmitter.prototype.off = function(ename, callback) {
    if (this.events[ename]) {
        // this.events[ename] && this.events[ename].filter(cb => cb != callback)
        // console.log('this.events', this.events)
        const targetIndex = this.events[ename].findIndex(cb => cb === callback)
        if (targetIndex !== -1) {
            this.events[ename].splice(targetIndex, 1)
        }
        if (this.events[ename].length === 0) {
            delete this.events[ename]
        }
    }
}
// 只执行一次订阅发布,然后移除
EventEmitter.prototype.once = function(ename, callback) {
    var that = this;
    var fn = function() {
        callback();
        that.off(ename, fn);
    }
    this.on(ename, fn);
}


// 实例化构造函数
var em = new EventEmitter();

em.on("work", function() {
    console.log('work,订阅发布成功');
});

var makeOnce = function() {
    console.log("money,移除");
}
em.on("money", makeOnce);

em.once("love", function() {
    console.log("love,仅一次");
});

em.emit("work");
em.off("money", makeOnce);
em.emit("money"); // 移除后,无法发布
em.emit("love"); 
em.emit("love"); // 只执行一次,再次将不再执行

2、如下是打印结果

相关推荐
j***89462 分钟前
spring-boot-starter和spring-boot-starter-web的关联
前端
star_11127 分钟前
Jenkins+nginx部署前端vue项目
前端·vue.js·jenkins
im_AMBER15 分钟前
Canvas架构手记 05 鼠标事件监听 | 原生事件封装 | ctx 结构化对象
前端·笔记·学习·架构
JIngJaneIL16 分钟前
农产品电商|基于SprinBoot+vue的农产品电商系统(源码+数据库+文档)
java·前端·数据库·vue.js·spring boot·毕设·农产品电商系统
Tongfront21 分钟前
前端通用submit方法
开发语言·前端·javascript·react
可爱又迷人的反派角色“yang”24 分钟前
LVS+Keepalived群集
linux·运维·服务器·前端·nginx·lvs
han_24 分钟前
前端高频面试题之CSS篇(二)
前端·css·面试
JIngJaneIL26 分钟前
书店销售|书屋|基于SprinBoot+vue书店销售管理设计与实现(源码+数据库+文档)
java·前端·数据库·vue.js·spring boot·毕设·书店销售管理设计与实现
徐同保27 分钟前
n8n CLI 项目结构全面分析(node后端)
前端
墨雪不会编程36 分钟前
C++基础语法篇五 ——类和对象
java·前端·c++