当前位置: 首页 > 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&…...

利用最小二乘法找圆心和半径

#include <iostream> #include <vector> #include <cmath> #include <Eigen/Dense> // 需安装Eigen库用于矩阵运算 // 定义点结构 struct Point { double x, y; Point(double x_, double y_) : x(x_), y(y_) {} }; // 最小二乘法求圆心和半径 …...

【Oracle APEX开发小技巧12】

有如下需求&#xff1a; 有一个问题反馈页面&#xff0c;要实现在apex页面展示能直观看到反馈时间超过7天未处理的数据&#xff0c;方便管理员及时处理反馈。 我的方法&#xff1a;直接将逻辑写在SQL中&#xff0c;这样可以直接在页面展示 完整代码&#xff1a; SELECTSF.FE…...

Unity3D中Gfx.WaitForPresent优化方案

前言 在Unity中&#xff0c;Gfx.WaitForPresent占用CPU过高通常表示主线程在等待GPU完成渲染&#xff08;即CPU被阻塞&#xff09;&#xff0c;这表明存在GPU瓶颈或垂直同步/帧率设置问题。以下是系统的优化方案&#xff1a; 对惹&#xff0c;这里有一个游戏开发交流小组&…...

《Playwright:微软的自动化测试工具详解》

Playwright 简介:声明内容来自网络&#xff0c;将内容拼接整理出来的文档 Playwright 是微软开发的自动化测试工具&#xff0c;支持 Chrome、Firefox、Safari 等主流浏览器&#xff0c;提供多语言 API&#xff08;Python、JavaScript、Java、.NET&#xff09;。它的特点包括&a…...

解锁数据库简洁之道:FastAPI与SQLModel实战指南

在构建现代Web应用程序时&#xff0c;与数据库的交互无疑是核心环节。虽然传统的数据库操作方式&#xff08;如直接编写SQL语句与psycopg2交互&#xff09;赋予了我们精细的控制权&#xff0c;但在面对日益复杂的业务逻辑和快速迭代的需求时&#xff0c;这种方式的开发效率和可…...

IoT/HCIP实验-3/LiteOS操作系统内核实验(任务、内存、信号量、CMSIS..)

文章目录 概述HelloWorld 工程C/C配置编译器主配置Makefile脚本烧录器主配置运行结果程序调用栈 任务管理实验实验结果osal 系统适配层osal_task_create 其他实验实验源码内存管理实验互斥锁实验信号量实验 CMISIS接口实验还是得JlINKCMSIS 简介LiteOS->CMSIS任务间消息交互…...

MySQL JOIN 表过多的优化思路

当 MySQL 查询涉及大量表 JOIN 时&#xff0c;性能会显著下降。以下是优化思路和简易实现方法&#xff1a; 一、核心优化思路 减少 JOIN 数量 数据冗余&#xff1a;添加必要的冗余字段&#xff08;如订单表直接存储用户名&#xff09;合并表&#xff1a;将频繁关联的小表合并成…...

第7篇:中间件全链路监控与 SQL 性能分析实践

7.1 章节导读 在构建数据库中间件的过程中&#xff0c;可观测性 和 性能分析 是保障系统稳定性与可维护性的核心能力。 特别是在复杂分布式场景中&#xff0c;必须做到&#xff1a; &#x1f50d; 追踪每一条 SQL 的生命周期&#xff08;从入口到数据库执行&#xff09;&#…...

PHP 8.5 即将发布:管道操作符、强力调试

前不久&#xff0c;PHP宣布了即将在 2025 年 11 月 20 日 正式发布的 PHP 8.5&#xff01;作为 PHP 语言的又一次重要迭代&#xff0c;PHP 8.5 承诺带来一系列旨在提升代码可读性、健壮性以及开发者效率的改进。而更令人兴奋的是&#xff0c;借助强大的本地开发环境 ServBay&am…...

何谓AI编程【02】AI编程官网以优雅草星云智控为例建设实践-完善顶部-建立各项子页-调整排版-优雅草卓伊凡

何谓AI编程【02】AI编程官网以优雅草星云智控为例建设实践-完善顶部-建立各项子页-调整排版-优雅草卓伊凡 背景 我们以建设星云智控官网来做AI编程实践&#xff0c;很多人以为AI已经强大到不需要程序员了&#xff0c;其实不是&#xff0c;AI更加需要程序员&#xff0c;普通人…...