有趣的Nodejs模块events

作者 : 开心源码 本文共2477个字,预计阅读时间需要7分钟 发布时间: 2022-05-13 共204人阅读

读了 events 模块的文档,研究了几个有意思的问题:

  • ??? 事件驱动模型
  • ??? 优雅的错误解决
  • ??? 监听器器队列顺序解决
  • ??? 内存管理与防止泄漏

引用/转载 请公告出处:原文链接: xxoo521.com

事件驱动模型

Nodejs 使用了一个事件驱动、非阻塞 IO 的模型。events模块是事件驱动的核心模块。很多内置模块都继承了events.EventEmitter

自己无需手动实现这种设计模式,直接继承EventEmitter就可。代码如下:

const { EventEmitter } = require("events");class MyEmitter extends EventEmitter {}const ins = new MyEmitter();ins.on("test", () => {    console.log("emit test event");});ins.emit("test");

优雅的错误解决

根据文档,应该 EventEmitter 实例的error事件是个特殊事件。推荐做法是:在创立实例后,应该立即注册error事件。

const ins = new MyEmitter();ins.on("error", error => {    console.log("error msg is", error.message);});

注册error事件后,我本来的了解是,所有事件回掉逻辑中的错误都会在 EventEmitter 内部被捕获,并且在内部触发 error 事件。

也就是说下面代码,会打印:”error msg is a is not defined”。

ins.on("test", () => {    console.log(a);});ins.emit("test");

然而,错误并没有捕获,直接抛出了异常。由此可见,EventEmitter 在执行内部逻辑的时候,并没有try-catch。这个起因,请见Node Issue。简单来讲,Error 和 Exception 并不完全一样。

假如按照正常想法,不想每一次都在外面套一层try-catch,那应该怎样做呢?我的做法是在
EventEmitter 原型链上新添加一个safeEmit函数。

EventEmitter.prototype.safeEmit = function(name, ...args) {    try {        return this.emit(name, ...args);    } catch (error) {        return this.emit("error", error);    }};

如此一来,运行前一段代码的 Exception 就会被捕获到,并且触发error事件。前一段代码的输出就变成了:

error msg is a is not defined

监听器队列顺序解决

对于同一个事件,触发它的时候,函数的执行顺序就是函数绑定时候的顺序。官方库提供了emitter.prependListener()emitter.prependOnceListener() 两个接口,可以让新的监听器直接增加到队列头部。

但是假如想让新的监听器放入任何监听器队列的任何位置呢?在原型链上封装了 insertListener 方法。

EventEmitter.prototype.insertListener = function(    name,    index,    callback,    once = false) {    // 假如是once监听器,其数据结构是 {listener: Function}    // 正常监听器,直接是 Function    const listeners = ins.rawListeners(name);    const that = this;    // 下标不合法    if (index > listeners.length || index < 0) {        return false;    }    // 绑定监听器数量已达上限    if (listeners.length >= this.getMaxListeners()) {        return false;    }    listeners.splice(index, 0, once ? { listener: callback } : callback);    this.removeAllListeners(name);    listeners.forEach(function(item) {        if (typeof item === "function") {            that.on(name, item);        } else {            const { listener } = item;            that.once(name, listener);        }    });    return true;};

使用起来,效果如下:

const ins = new MyEmitter();ins.on("error", error => {    console.log("error msg is", error.message);});ins.on("test", () => {    console.log("test 1");});ins.on("test", () => {    console.log("test 2");});// 监听器队列中插入新的监听器,一个是once类型,一个不是once类型ins.insertListener(    "test",    0,    () => {        console.log("once test insert");    },    true);ins.insertListener("test", 1, () => {    console.log("test insert");});

连续调用两次ins.emit("test"),结果输出如下:

# 第一次once test inserttest inserttest 1test 2# 第二次: once 类型的监听器调用一次后销毁test inserttest 1test 2

内存管理与防止泄漏

在绑定事件监听器的时候,假如监听器没有被 remove,那么存在内存泄漏的风险。

我知道的常见做法如下:

  • 经常 CR,移除不需要的事件监听器
  • 通过once绑定监听器,调用一次后,监听器被自动移除
  • [推荐]hack 一个更安全的EventEmitter

参考链接

  • NodeJS Issue
  • Docs: events

专注前台与算法的系列干货分享,欢迎关注(???)

image

说明
1. 本站所有资源来源于用户上传和网络,如有侵权请邮件联系站长!
2. 分享目的仅供大家学习和交流,您必须在下载后24小时内删除!
3. 不得使用于非法商业用途,不得违反国家法律。否则后果自负!
4. 本站提供的源码、模板、插件等等其他资源,都不包含技术服务请大家谅解!
5. 如有链接无法下载、失效或广告,请联系管理员处理!
6. 本站资源售价只是摆设,本站源码仅提供给会员学习使用!
7. 如遇到加密压缩包,请使用360解压,如遇到无法解压的请联系管理员
开心源码网 » 有趣的Nodejs模块events

发表回复