当前位置: 首页 > news >正文

Node.js 中的 Event 模块详解

Node.js 中的 Event 模块是实现事件驱动编程的核心模块。它基于观察者模式,允许对象(称为“事件发射器”)发布事件,而其他对象(称为“事件监听器”)可以订阅并响应这些事件。这种模式非常适合处理异步操作和事件驱动的场景。


1. 概念

1.1 事件驱动编程

事件驱动编程是一种编程范式,程序的执行流程由事件(如用户输入、文件读取完成、网络请求响应等)决定。Node.js 的核心设计理念就是基于事件驱动的非阻塞 I/O 模型。

1.2 事件发射器(EventEmitter)

EventEmitter 是 Node.js 中实现事件驱动编程的核心类。它提供了以下功能:

  • 发布事件:通过 emit() 方法触发事件。
  • 订阅事件:通过 on()addListener() 方法监听事件。
  • 取消订阅:通过 removeListener()off() 方法移除事件监听器。

2. 定义与用法

2.1 引入 EventEmitter

EventEmitterevents 模块的一个类,使用前需要引入:

const EventEmitter = require('events');

2.2 创建事件发射器

可以通过继承 EventEmitter 或直接实例化来创建事件发射器。

方法 1:直接实例化
const EventEmitter = require('events');// 创建事件发射器实例
const myEmitter = new EventEmitter();// 监听事件
myEmitter.on('greet', (name) => {console.log(`Hello, ${name}!`);
});// 触发事件
myEmitter.emit('greet', 'Alice'); // 输出:Hello, Alice!
方法 2:继承 EventEmitter
const EventEmitter = require('events');// 自定义类继承 EventEmitter
class MyEmitter extends EventEmitter {}// 创建自定义类的实例
const myEmitter = new MyEmitter();// 监听事件
myEmitter.on('greet', (name) => {console.log(`Hello, ${name}!`);
});// 触发事件
myEmitter.emit('greet', 'Bob'); // 输出:Hello, Bob!

2.3 常用方法

1. on(eventName, listener)
  • 监听指定事件。
  • eventName:事件名称。
  • listener:事件触发时的回调函数。
myEmitter.on('data', (data) => {console.log('Data received:', data);
});
2. emit(eventName[, ...args])
  • 触发指定事件。
  • eventName:事件名称。
  • args:传递给监听器的参数。
myEmitter.emit('data', { message: 'Hello, world!' });
3. once(eventName, listener)
  • 监听事件,但只触发一次。
  • 触发后自动移除监听器。
myEmitter.once('init', () => {console.log('Initialized!');
});myEmitter.emit('init'); // 输出:Initialized!
myEmitter.emit('init'); // 无输出
4. removeListener(eventName, listener)
  • 移除指定事件的监听器。
const listener = (data) => {console.log('Data received:', data);
};myEmitter.on('data', listener);
myEmitter.removeListener('data', listener);
5. off(eventName, listener)
  • removeListener 的别名,功能相同。
6. removeAllListeners([eventName])
  • 移除所有监听器,或指定事件的所有监听器。
myEmitter.removeAllListeners('data');
7. listenerCount(eventName)
  • 返回指定事件的监听器数量。
const count = myEmitter.listenerCount('data');
console.log('Listener count:', count);

3. 优缺点

3.1 优点

  1. 解耦
    • 事件驱动模式将事件的发布和订阅解耦,使代码更模块化和可维护。
  2. 异步支持
    • 非常适合处理异步操作,如文件 I/O、网络请求等。
  3. 灵活性
    • 可以动态添加或移除事件监听器,适应不同的业务需求。
  4. 内置支持
    • Node.js 的许多核心模块(如 fsnethttp)都基于 EventEmitter

3.2 缺点

  1. 回调地狱
    • 如果事件嵌套过多,可能会导致回调地狱,降低代码可读性。
  2. 错误处理
    • 如果没有正确监听 error 事件,可能会导致程序崩溃。
  3. 内存泄漏
    • 如果未及时移除监听器,可能会导致内存泄漏。
  4. 调试困难
    • 事件驱动的代码流程不如同步代码直观,调试起来可能更复杂。

4. 最佳实践

4.1 错误处理

始终监听 error 事件,避免未捕获的错误导致程序崩溃。

myEmitter.on('error', (err) => {console.error('Error occurred:', err.message);
});myEmitter.emit('error', new Error('Something went wrong!'));

4.2 避免内存泄漏

及时移除不再需要的监听器。

const listener = () => {console.log('Event triggered');
};myEmitter.on('event', listener);// 移除监听器
myEmitter.off('event', listener);

4.3 使用 once 替代 on

如果事件只需要触发一次,使用 once 而不是 on,避免手动移除监听器。

myEmitter.once('init', () => {console.log('Initialized!');
});

5. 示例:文件读取事件

以下是一个结合 fs 模块的文件读取示例:

const fs = require('fs');
const EventEmitter = require('events');class FileReader extends EventEmitter {readFile(filePath) {fs.readFile(filePath, 'utf8', (err, data) => {if (err) {this.emit('error', err);} else {this.emit('data', data);}});}
}const reader = new FileReader();reader.on('data', (data) => {console.log('File content:', data);
});reader.on('error', (err) => {console.error('Failed to read file:', err.message);
});reader.readFile('example.txt');

6. 总结

  • EventEmitter 是 Node.js 中实现事件驱动编程的核心工具。
  • 优点:解耦、异步支持、灵活性高。
  • 缺点:回调地狱、错误处理复杂、可能内存泄漏。
  • 适用场景:异步操作、事件驱动的应用(如服务器、文件 I/O 等)。

通过合理使用 EventEmitter,可以编写出高效、模块化的 Node.js 应用程序。

相关文章:

Node.js 中的 Event 模块详解

Node.js 中的 Event 模块是实现事件驱动编程的核心模块。它基于观察者模式,允许对象(称为“事件发射器”)发布事件,而其他对象(称为“事件监听器”)可以订阅并响应这些事件。这种模式非常适合处理异步操作和…...

EasyRTC嵌入式WebRTC视频通话SDK支持Web浏览器、Linux、ARM、Android、iOS

随着互联网技术的飞速发展,实时通信(RTC)已经成为现代应用中不可或缺的一部分。无论是视频会议、在线教育、远程医疗,还是社交娱乐,实时通信技术都在其中扮演着重要角色。 然而,WebRTC技术在PC和移动端的支…...

pycharm社区版有个window和arm64版本,到底下载哪一个?还有pycharm官网

首先pycharm官网是这一个。我是在2025年2月16日9:57进入的网站。如果网站还没有更新的话,那么就往下滑一下找到 community Edition,这个就是社区版了免费的。PyCharm:适用于数据科学和 Web 开发的 Python IDE 适用于数据科学和 Web 开发的 Python IDE&am…...

【玩转全栈】----Django模板语法、请求与响应

目录 一、引言 二、模板语法 三、传参 1、视图函数到模板文件 2、模板文件到视图函数 四、引入静态文件 五、请求与响应 ?1、请求 2、响应 六、综合小案例 1、源码展示 2、注意事项以及部分解释 3、展示 一、引言 像之前那个页面,太过简陋,而且一个完整…...

网络安全:挑战、技术与未来发展

📝个人主页🌹:一ge科研小菜鸡-CSDN博客 🌹🌹期待您的关注 🌹🌹 1. 引言 在数字化时代,网络安全(Cybersecurity)已成为全球关注的焦点。随着云计算、大数据、…...

DeepSeek 服务器繁忙的全面解决方案

目录 引文 正文 一、 服务器繁忙的原因分析 二、 解决方案 2.1切换网络 2.2使用网络加速工具 2.3错峰使用DeepSeek 2.4本地部署 2.5调用API 三、官方动态 一、技术研发与产品升级 二、市场合作与商业化进展 三、区域化布局与产业赋能 四、未来规划与社会责任 结语…...

将OpenWrt部署在x86服务器上

正文共:1234 字 40 图,预估阅读时间:2 分钟 如果你问ChatGPT有哪些开源的SD-WAN方案,他会这样答复你: 我们看到,OpenWrt也属于比较知名的开源SD-WAN解决方案。当然,在很久之前,我就发…...

计算机视觉:卷积神经网络(CNN)基本概念(一)

第一章:计算机视觉中图像的基础认知 第二章:计算机视觉:卷积神经网络(CNN)基本概念(一) 第三章:计算机视觉:卷积神经网络(CNN)基本概念(二) 第四章:搭建一个经典的LeNet5神经网络 一、引言 卷积神经网络&…...

企业文件共享中的权限管理与安全风险防范

在企业的日常运营中,文件共享是必不可少的一项工作。然而,文件共享过程中如果权限管理不当,极易引发安全风险,导致企业敏感信息泄露。因此,加强文件共享中的权限管理与安全风险防范,对于保障企业信息安全至…...

使用DeepSeek建立一个智能聊天机器人0.12

为了确保这段代码能够在Windows和Linux系统上都能正常运行,我考虑以下几个方面: 路径分隔符:在Windows和Linux中,文件路径的分隔符不同。Windows使用反斜杠(\),而Linux使用正斜杠(/)。我们可以使用 os.path.join 来处理路径,以确保跨平台兼容性。 消息框:tkinter.…...

国家队出手!DeepSeek上线国家超算互联网平台!

目前,国家超算互联网平台已推出 DeepSeek – R1 模型的 1.5B、7B、8B、14B 版本,后续还会在近期更新 32B、70B 等版本。 DeepSeek太火爆了!在这个春节档,直接成了全民热议的话题。 DeepSeek也毫无悬念地干到了全球增速最快的AI应用。这几天,国内的云计算厂家都在支持Dee…...

Deep seek学习日记1

Deepseek最强大的就是它的深度思考,并且展现了它的思考过程。 五种可使用Deep seek的方式(应该不限于这五种,后续嵌入deepseek的应该更多,多了解一点因为官网容易崩~~): 1.deep seek官网 2.硅基流动silicon…...

乐理笔记(持续更新)

单音与音程 单音:由一个音组成。 音程:由两个音组成,表示两个音之间的音高距离。 如何数音程: 单音程:9 - X,性质相反。例如,9度音程减去某个数,性质会相反。 复音程&#xff1a…...

【动态路由】系统Web URL资源整合系列(后端技术实现)【nodejs实现】

需求说明 软件功能需求:反向代理功能(描述:apollo、eureka控、apisix、sentinel、普米、kibana、timetask、grafana、hbase、skywalking-ui、pinpoint、cmak界面、kafka-map、nacos、gateway、elasticsearch、 oa-portal 业务应用等多个web资…...

PHP高效、轻量级表格数据处理库 OpenSpout ,很好用

OpenSpout 是一个高效、轻量级的 PHP 库,用于处理电子表格文件(如 Excel 和 CSV)。它支持读取和写入大型文件,且内存占用低。本文将详细介绍如何安装和使用 OpenSpout。 目录 安装 基本使用 高级功能 参考文档 安装 OpenSp…...

2010年上半年软件设计师考试上午真题的知识点整理(附真题及答案解析)

以下是2010年上半年软件设计师考试上午真题的知识点分类整理,涉及定义的详细解释,供背诵记忆。 1. 计算机组成原理 CPU与存储器的访问。 Cache的作用: 提高CPU访问主存数据的速度,减少访问延迟。存储器的层次结构: 包括寄存器、Cache、主存和…...

EventSource的使用

什么是EventSource EventSource 是一个用于服务器推送事件(Server-Sent Events, SSE)的接口,它允许服务器推送实时更新到浏览器。与 WebSocket 不同,SSE 是单向的(服务器到客户端),适用于更新频…...

【第12章:深度学习与伦理、隐私—12.3 深度学习模型的透明性与可解释性提升策略】

凌晨三点的ICU病房,AI辅助诊断系统将一位患者的肺炎误判为普通感冒——当主治医生要求查看诊断依据时,系统只能给出冷冰冰的概率数值。这场惊心动魄的误诊事件,掀开了深度学习可解释性危机的冰山一角。 一、模型透明的"第一性原理" 1.1 可解释性的三维度量 ![可…...

RocketMq中RouteInfoManger组件的源码分析

1.前言 RouteInfoManager 是 RocketMQ 中 NameServer 的核心组件之一,主要负责管理和维护整个 RocketMQ 集群的路由元数据信息。里面包含一些非常核心的功能:存储和管理 Broker 信息(broker的注册,broker心跳的维护)&…...

java八股文-mysql

1. 索引 1.1 什么是索引 索引(index)是帮助Mysql高效获取数据的数据结构(有序).提高数据的检索效率,降低数据库的IO成本(不需要全表扫描).通过索引列对数据进行排序,降低数据排序成本,降低了CPU的消耗. 1.2 mysql索引使用的B树? 1. 没有使用二叉树,最坏情况o&…...

挑战杯推荐项目

“人工智能”创意赛 - 智能艺术创作助手:借助大模型技术,开发能根据用户输入的主题、风格等要求,生成绘画、音乐、文学作品等多种形式艺术创作灵感或初稿的应用,帮助艺术家和创意爱好者激发创意、提高创作效率。 ​ - 个性化梦境…...

国防科技大学计算机基础课程笔记02信息编码

1.机内码和国标码 国标码就是我们非常熟悉的这个GB2312,但是因为都是16进制,因此这个了16进制的数据既可以翻译成为这个机器码,也可以翻译成为这个国标码,所以这个时候很容易会出现这个歧义的情况; 因此,我们的这个国…...

生成xcframework

打包 XCFramework 的方法 XCFramework 是苹果推出的一种多平台二进制分发格式,可以包含多个架构和平台的代码。打包 XCFramework 通常用于分发库或框架。 使用 Xcode 命令行工具打包 通过 xcodebuild 命令可以打包 XCFramework。确保项目已经配置好需要支持的平台…...

CTF show Web 红包题第六弹

提示 1.不是SQL注入 2.需要找关键源码 思路 进入页面发现是一个登录框,很难让人不联想到SQL注入,但提示都说了不是SQL注入,所以就不往这方面想了 ​ 先查看一下网页源码,发现一段JavaScript代码,有一个关键类ctfs…...

React Native 开发环境搭建(全平台详解)

React Native 开发环境搭建(全平台详解) 在开始使用 React Native 开发移动应用之前,正确设置开发环境是至关重要的一步。本文将为你提供一份全面的指南,涵盖 macOS 和 Windows 平台的配置步骤,如何在 Android 和 iOS…...

系统设计 --- MongoDB亿级数据查询优化策略

系统设计 --- MongoDB亿级数据查询分表策略 背景Solution --- 分表 背景 使用audit log实现Audi Trail功能 Audit Trail范围: 六个月数据量: 每秒5-7条audi log,共计7千万 – 1亿条数据需要实现全文检索按照时间倒序因为license问题,不能使用ELK只能使用…...

Nuxt.js 中的路由配置详解

Nuxt.js 通过其内置的路由系统简化了应用的路由配置,使得开发者可以轻松地管理页面导航和 URL 结构。路由配置主要涉及页面组件的组织、动态路由的设置以及路由元信息的配置。 自动路由生成 Nuxt.js 会根据 pages 目录下的文件结构自动生成路由配置。每个文件都会对…...

零基础设计模式——行为型模式 - 责任链模式

第四部分:行为型模式 - 责任链模式 (Chain of Responsibility Pattern) 欢迎来到行为型模式的学习!行为型模式关注对象之间的职责分配、算法封装和对象间的交互。我们将学习的第一个行为型模式是责任链模式。 核心思想:使多个对象都有机会处…...

c#开发AI模型对话

AI模型 前面已经介绍了一般AI模型本地部署,直接调用现成的模型数据。这里主要讲述讲接口集成到我们自己的程序中使用方式。 微软提供了ML.NET来开发和使用AI模型,但是目前国内可能使用不多,至少实践例子很少看见。开发训练模型就不介绍了&am…...

【JavaSE】绘图与事件入门学习笔记

-Java绘图坐标体系 坐标体系-介绍 坐标原点位于左上角,以像素为单位。 在Java坐标系中,第一个是x坐标,表示当前位置为水平方向,距离坐标原点x个像素;第二个是y坐标,表示当前位置为垂直方向,距离坐标原点y个像素。 坐标体系-像素 …...