如何实现一个符合ECMAScript规范的EventEmitter类?

答案是实现一个轻量且符合 Node.js 规范的 EventEmitter 类,核心包括:使用 _events 存储事件映射;提供 on/addListener 注册监听器;emit 按序触发并处理异常;off/removeListener 移除指定监听器;once 实现一次性监听;支持 removeAllListeners、eventNames、listenerCount 等扩展方法,确保链式调用与类型安全,避免内存泄漏。

如何实现一个符合ecmascript规范的eventemitter类?

要实现一个符合 ECMAScript 规范、轻量且功能完整的 EventEmitter 类,核心是管理事件监听器的注册、触发和移除。虽然 ECMAScript 标准本身不包含 EventEmitter,但 Node.js 的 EventEmitter 模式已成为事实标准。我们可以依据其行为来实现一个兼容的版本。

1. 基本结构与事件存储

使用一个对象或 Map 来存储事件名到回调函数数组的映射。每个事件可以有多个监听器。

class EventEmitter {
  constructor() {
    this._events = Object.create(null); // 避免原型污染
  }
}

2. 实现 on 和 addListener(别名)

将监听器添加到指定事件的回调列表中。支持重复添加相同监听器。

说明: on 与 addListener 是同一个方法的不同名称,应指向同一函数。

on(event, listener) {
  if (typeof listener !== ‘function’) {
    throw new TypeError(‘The listener must be a function’);
  }

  if (!this._events[event]) {
    this._events[event] = [];
  }

  this._events[event].push(listener);
  return this; // 支持链式调用
}

addListener(event, listener) {
  return this.on(event, listener);
}

3. 实现 emit:触发事件

按顺序执行指定事件的所有监听器,并传入参数。注意处理监听器执行期间可能抛出的错误。

emit(event, …args) {
  if (!this._events[event] || this._events[event].length === 0) {
    return false; // 无监听器返回 false
  }

  // 创建副本防止遍历中修改数组导致问题
  const listeners = this._events[event].slice();
  for (const listener of listeners) {
    listener.apply(this, args);
  }

  return true;
}

4. 实现 off 和 removeListener

从事件队列中移除指定监听器。注意只移除第一个匹配项。

off(event, listener) {
  if (typeof listener !== ‘function’) {
    throw new TypeError(‘The listener must be a function’);
  }

  const listeners = this._events[event];
  if (!listeners) return this;

  const index = listeners.indexOf(listener);
  if (index > -1) {
    listeners.splice(index, 1);
    // 清理空数组
    if (listeners.length === 0) {
      delete this._events[event];
    }
  }

  return this;
}

removeListener(event, listener) {
  return this.off(event, listener);
}

5. 支持 once:一次性监听器

监听器执行一次后自动移除。关键在于包装原始监听器。

once(event, listener) {
  const wrapped = (…args) => {
    this.off(event, wrapped);
    listener.apply(this, args);
  };
  // 存储原始引用,便于 off 移除
  wrapped.listener = listener;

  this.on(event, wrapped);
  return this;
}

6. 其他常用方法(可选增强)

removeAllListeners(event?):清除某个或所有事件的监听器eventNames():返回当前所有注册的事件名listenerCount(event):返回某事件的监听器数量rawListeners(event):返回某事件的监听器数组(含 once 包装)removeAllListeners(event) {
  if (event) {
    delete this._events[event];
  } else {
    this._events = Object.create(null);
  }
  return this;
}

eventNames() {
  return Object.keys(this._events);
}

listenerCount(event) {
  return this._events[event] ? this._events[event].length : 0;
}

基本上就这些。这个实现覆盖了 EventEmitter 的核心行为,符合 Node.js 的 API 设计,同时保持代码简洁、类型安全,并避免常见陷阱如异步执行、监听器泄漏等。实际项目中可根据需要扩展更多功能,比如最大监听器警告(setMaxListeners)。

以上就是如何实现一个符合ECMAScript规范的EventEmitter类?的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1523287.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月20日 15:47:12
下一篇 2025年12月20日 15:47:28

相关推荐

  • React Context中Ref元素事件监听的陷阱与focusout的妙用

    本文探讨了在React应用中,通过Context传递的DOM引用(ref)上监听blur事件时可能遇到的问题。核心在于blur事件不冒泡,导致在父元素上无法捕获子元素的失焦行为。解决方案是使用focusout事件,它具备冒泡特性,能有效处理此类场景,确保事件监听器按预期触发。 在react开发中,我…

    2025年12月20日
    000
  • JavaScript中JSON数据重构:将动态日期键转换为结构化对象

    在现代Web开发中,处理和重构JSON数据是常见的任务。有时,我们面临的数据结构中包含动态生成的键,例如日期,这使得直接操作和分析变得复杂。本教程将指导您如何在JavaScript中重构JSON数据。当原始数据包含动态日期作为键,且需要将其转换为以日期为核心、教育类型为字段的结构时,我们将通过识别所…

    2025年12月20日
    000
  • 如何构建一个支持多主题切换的CSS架构?

    实现多主题切换的关键在于使用CSS自定义属性定义主题样式,并通过data-theme属性与JavaScript动态控制外观,结合localStorage保存用户偏好,可选按需加载CSS文件优化性能,适用于各类现代前端框架。 实现多主题切换的关键在于结构清晰、可维护性强的CSS架构。核心思路是将主题样…

    2025年12月20日
    000
  • 什么是 JavaScript 的装饰器在元编程中的应用,它们如何修改类的行为?

    装饰器是一种元编程特性,通过@语法修饰类或成员,实现行为扩展。它分为类、方法、属性等类型,可添加静态属性、日志记录、权限控制等功能,如@addVersion为类添加version属性,@log拦截方法调用输出参数信息,广泛应用于依赖注入、缓存、序列化等场景,提升代码复用与维护性,但仅适用于类上下文,…

    2025年12月20日
    000
  • Web应用中安全实现用户会话持久化:JWT认证指南

    在为Discord Bot构建仪表盘时,实现用户登录状态在页面刷新后依然保持是一项常见需求。本文将探讨传统方法(如直接使用localStorage或基于IP的验证)的安全性隐患,并重点介绍如何利用JSON Web Tokens (JWT) 这一行业标准,以加密签名的方式安全地管理用户会话,确保身份验…

    2025年12月20日
    000
  • JavaScript中实现表单提交前的确认提示与取消机制

    本教程详细讲解如何在JavaScript中为HTML表单添加提交前的确认对话框。通过监听submit事件并利用event.preventDefault()方法,我们能有效控制表单的提交流程,确保用户在关键操作前进行二次确认,并在用户选择取消时阻止表单提交,从而提升用户体验和数据准确性。 在Web开发…

    2025年12月20日
    000
  • JavaScript map 方法中函数闭包变量捕获机制详解

    本文深入探讨了JavaScript map 方法中,匿名函数内部变量捕获与闭包的机制。针对在 map 迭代过程中,函数定义中引用的外部变量(如 item.type)未在日志输出中“替换”为实际值的问题,文章阐明了这是对函数定义与执行、以及闭包工作原理的常见误解。通过示例代码,详细演示了变量在函数创建…

    2025年12月20日
    000
  • 如何构建一个微前端架构下的JavaScript应用?

    微前端通过按路由拆分子应用,选用qiankun实现隔离与通信,主应用统一管理依赖与状态,提升系统可维护性与团队协作效率。 构建一个微前端架构下的 JavaScript 应用,核心在于将大型前端项目拆分为多个独立、可自治的子应用,这些子应用可以由不同团队使用不同的技术栈开发,并能独立部署和运行。关键不…

    2025年12月20日
    000
  • 如何构建一个不依赖框架的、渐进增强的客户端路由系统?

    先保证链接可访问和页面跳转,再用 JavaScript 增强体验。通过原生 History API(pushState、replaceState)更新 URL 并监听 popstate 事件实现无刷新路由,拦截内链点击进行异步内容加载;结合路由表匹配路径并渲染对应视图,支持动态插入 HTML 或懒加…

    2025年12月20日 好文分享
    000
  • 如何用JavaScript实现自然语言处理的基础功能?

    JavaScript可通过原生方法和库实现基础NLP功能:1. 使用split或nodejieba进行中英文分词;2. 借助停用词表过滤无意义词汇;3. 通过freqMap统计词频并提取关键词;4. 利用Compromise、Natural等库增强分析能力,适用于前端轻量级处理。 JavaScrip…

    2025年12月20日
    000
  • 如何实现一个类型安全的Event Emitter?

    答案是使用泛型和索引类型实现类型安全的 Event Emitter。通过定义 Events 接口明确事件名与参数类型,结合 TypedEmitter 泛型类约束 on、emit 方法的事件名和参数类型,确保编译时检查正确性,避免拼写错误或参数不匹配问题,提升代码健壮性。 实现一个类型安全的 Even…

    2025年12月20日
    000
  • JavaScript类中的公共实例字段:深入理解其工作原理与原型链的关系

    本文深入探讨JavaScript ES6类中公共实例字段(Public Instance Fields)的内部工作机制。揭示这些字段并非存储在类的原型链上,而是直接在每个实例创建时通过构造函数赋值,从而解释了为何它们不能通过原型链访问,并强调了它们作为实例独有属性的特性。 在javascript中,…

    2025年12月20日
    000
  • 移动设备上自定义下拉列表不显示的解决方案:HTML结构与JS渲染指南

    针对WordPress插件中自定义自动完成下拉列表在移动设备上无法显示的问题,本文深入分析了常见的HTML结构误用,特别是在JavaScript动态生成下拉选项时,将元素错误地嵌套在 而非中导致渲染失败。文章提供了详细的解决方案,通过修改JavaScript代码确保生成正确的标签结构,从而解决移动设…

    2025年12月20日
    000
  • 移动端自动完成下拉列表显示异常:HTML语义化与iOS兼容性修复

    本文探讨了JavaScript动态生成的自动完成下拉列表在移动设备(尤其是iOS)上不显示的问题。通过分析发现,问题根源在于使用非语义化的 元素来承载标签,而非标准的元素。文章将详细解释此兼容性问题的原因,并提供正确的HTML结构和JavaScript代码修改方案,以确保下拉列表在各类移动设备上正常…

    2025年12月20日
    000
  • ECharts旭日图:实现点击父节点动态显示/隐藏子节点

    本教程详细阐述如何在ECharts旭日图中实现点击父节点动态显示或隐藏其子节点的交互功能。通过禁用默认的节点点击行为,结合ECharts的事件监听机制和setOption方法,我们引入一个自定义的hidden_children数据属性来管理子节点的可见性。当用户点击特定父节点时,该节点下的子节点将根…

    2025年12月20日
    000
  • 解决 npm ERR! code ENOENT 错误:React 项目创建指南

    在创建 React 项目时,开发者常会遇到 npm ERR! code ENOENT 错误,这通常表示 npm 无法找到某个文件或目录。本教程将深入解析此错误,并提供一个核心解决方案:手动创建缺失的 AppDataRoamingnpm 目录,同时探讨其他潜在原因及排查方法,确保您能顺利启动 Reac…

    2025年12月20日
    000
  • 如何利用 JavaScript 实现一个基于事件溯源的事件存储系统?

    事件溯源通过记录状态变化为不可变事件流实现状态管理,使用JavaScript可构建轻量级系统。首先定义包含类型、时间、数据和聚合ID的事件结构,并用数组模拟事件存储;接着创建聚合根如BankAccount类,通过applyEvent方法根据事件类型更新状态,并提供deposit、withdraw等行…

    2025年12月20日
    000
  • Web应用安全登录:基于JWT实现用户会话持久化

    本文探讨了在Discord Bot仪表盘等Web应用中,如何安全地实现用户登录状态的持久化,避免每次刷新页面都重新登录。针对localStorage的安全性缺陷和IP地址存储的局限性,重点介绍了JSON Web Token (JWT) 作为一种基于加密签名的解决方案,确保用户身份验证的安全性与会话的…

    2025年12月20日
    000
  • 解决 npx 运行时 npm ERR! code ENOENT 错误

    当执行 npx 命令(如 create-react-app)时,若遇到 npm ERR! code ENOENT 错误,这通常表示 npm 无法找到其操作所需的某个文件或目录。本文将详细解析此错误,并提供一种常见的解决方案:通过手动创建缺失的 npm 目录来恢复 npm 的正常功能。 问题概述:np…

    2025年12月20日
    000
  • JavaScript控制表单提交:使用confirm对话框进行用户确认

    本教程详细介绍了如何使用JavaScript在HTML表单提交前添加用户确认对话框。通过监听submit事件并结合confirm()函数,开发者可以根据用户选择(确定或取消)来控制表单的提交行为,有效防止误操作,提升用户体验。文章提供了具体的代码示例和实现步骤。 在网页开发中,为了防止用户误操作或在…

    2025年12月20日
    000

发表回复

登录后才能评论
关注微信