javascript如何实现设计模式_单例模式和观察者模式如何写

单例模式确保类唯一实例并提供全局访问,核心是延迟初始化与实例缓存;观察者模式实现一对多依赖通知,含Subject与Observer角色,需注意内存泄漏与取消订阅。

javascript如何实现设计模式_单例模式和观察者模式如何写

单例模式确保一个类只有一个实例,并提供全局访问点;观察者模式定义对象间一对多依赖,当一个对象状态改变,所有依赖者自动收到通知。两者在 JavaScript 中实现简洁但需注意细节。

单例模式:用闭包或类+静态属性控制实例唯一性

核心是“延迟初始化 + 实例缓存”。避免每次 new 都创建新对象。

传统闭包写法(兼容性好):

const Singleton = (function() {
  let instance;
  function createInstance() {
    return {
      data: [],
      add(item) { this.data.push(item); },
      get() { return this.data; }
    };
  }
  return {
    getInstance() {
      if (!instance) {
        instance = createInstance();
      }
      return instance;
    }
  };
})();
// 使用
const a = Singleton.getInstance();
const b = Singleton.getInstance();
console.log(a === b); // true
ES6 Class 写法(更现代):
class Logger {
  static #instance;
  constructor() {
    if (Logger.#instance) {
      throw new Error(“Already instantiated”);
    }
    Logger.#instance = this;
  }
  static getInstance() {
    if (!Logger.#instance) {
      new Logger();
    }
    return Logger.#instance;
  }
  log(msg) { console.log(“[LOG]”, msg); }
}
// 使用
const logger1 = Logger.getInstance();
const logger2 = Logger.getInstance();
console.log(logger1 === logger2); // true

观察者模式:手动实现发布-订阅机制

关键角色:Subject(被观察者)、Observer(观察者)。不依赖第三方库,纯 JS 就能搭出轻量通信总线。

基础版 Subject 类:
class EventTarget {
  constructor() {
    this.observers = new Map();
  }
  subscribe(event, callback) {
    if (!this.observers.has(event)) {
      this.observers.set(event, new Set());
    }
    this.observers.get(event).add(callback);
  }
  unsubscribe(event, callback) {
    if (this.observers.has(event)) {
      this.observers.get(event).delete(callback);
    }
  }
  notify(event, data) {
    if (this.observers.has(event)) {
      for (const cb of this.observers.get(event)) {
        cb(data);
      }
    }
  }
}
// 使用示例
const bus = new EventTarget();
bus.subscribe(“user-login”, (user) => console.log(“欢迎”, user.name));
bus.subscribe(“user-login”, (user) => console.log(“更新头像”));
bus.notify(“user-login”, { name: “张三” }); 进阶建议:
– 用 WeakMap 存 observer 可避免内存泄漏(尤其 DOM 元素作为 observer 时)
– 加入 once 方法支持一次性监听
– notify 支持异步(如 Promise.all)避免阻塞主线程

实际用在哪?别硬套,看场景

单例适合:全局配置管理、日志器、状态仓库(如简易版 store)、WebSocket 连接实例。
观察者适合:组件通信(尤其无框架时)、表单联动、事件解耦(比如按钮点击后多个模块响应)、状态变更广播。

基本上就这些。两种模式都不复杂,但容易忽略边界情况——比如单例没防 new,观察者没做取消订阅,上线后可能悄悄吃内存。

立即学习“Java免费学习笔记(深入)”;

以上就是javascript如何实现设计模式_单例模式和观察者模式如何写的详细内容,更多请关注创想鸟其它相关文章!

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

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月21日 13:28:17
下一篇 2025年12月21日 13:28:28

相关推荐

  • Vue组件独立状态管理:解决多实例联动问题

    本文旨在解决vue.js应用中多个相同组件实例状态联动的问题。我们将探讨如何在父组件中通过独立的状态变量或状态数组,以及如何利用精确的事件处理机制(包括独立事件处理器或传递唯一标识符),确保每个组件实例能够独立地显示、隐藏和响应用户交互,从而实现组件的真正独立控制。 理解多组件实例联动问题 在Vue…

    2025年12月21日
    000
  • MUI X Date Picker 设置默认年份值:提升数据录入效率的实践指南

    本教程详细介绍了如何在mui x date picker组件中设置一个默认的年份值,以提高用户数据录入效率。通过利用`defaultvalue`属性并结合`dayjs`库,开发者可以轻松地将日期选择器预设为特定年份,例如2023年,从而优化用户体验,尤其适用于需要频繁输入同一年份数据的场景。 引言:…

    2025年12月21日
    000
  • MUI X DatePicker 设置默认年份值教程

    本教程详细介绍了如何在mui x的日期选择器中设置一个默认的年份值,以提高数据录入效率。通过利用`defaultvalue`属性和`dayjs`库,开发者可以轻松地将日期选择器初始化为指定年份,同时仍允许用户进行修改,从而优化特定业务场景下的用户体验。 在许多业务场景中,用户需要频繁录入大量数据,其…

    2025年12月21日
    000
  • JavaScript字符串中日期范围的提取与多格式转换

    本文详细介绍了如何使用JavaScript高效地从特定格式的字符串中提取日期范围,并将其转换为多种目标格式(YYYY-MM-DD和YYYYMM)。通过结合正则表达式进行初始匹配和自定义函数进行格式化,我们能够将原始日期字符串(如DD/MM/YYYY)转换为结构化的日期表示,最终生成包含起始和结束日期…

    2025年12月21日
    000
  • JavaScript中数字集合的字符包含关系检查教程

    本教程旨在详细阐述如何在javascript中高效地检查一个数字集合(winarray)中的元素是否以特定方式存在于另一个数字集合(mergeuserarray)的元素中。文章将深入探讨两种主要的匹配逻辑:无序数字包含(即所有组成数字是否存在)和有序子串匹配,并提供清晰的代码实现、应用场景及注意事项…

    2025年12月21日
    000
  • 条件语句 if-else if-else 的执行机制详解

    条件语句 `if-else if-else` 语句用于根据不同条件执行不同的代码块。其核心机制是顺序评估:系统会从上到下依次检查每个 `if` 和 `else if` 的条件。一旦找到第一个满足(即为真)的条件,对应的代码块就会被执行,并且整个条件链条随即终止。最终的 `else` 语句作为一个默认…

    好文分享 2025年12月21日
    000
  • 深入理解 Fetch API:正确解析 HTTP 响应数据

    fetch api 是现代 web 开发中用于进行网络请求的核心工具。本文将详细探讨 fetch 请求后如何正确解析不同类型的 http 响应体,包括文本、json 和二进制数据。我们将重点解决常见的响应体解析误区,特别是异步处理和一次性读取的特性,并通过实际代码示例指导读者高效地获取并处理服务器返…

    2025年12月21日
    000
  • React 父子组件间数组状态管理的最佳实践:实现子组件操作父组件数据过滤

    本教程探讨react父子组件间数组状态管理的有效方法。针对子组件触发操作并更新父组件中数组的需求,我们首先分析了直接在子组件中管理状态的不足。随后,介绍了通过将父组件的状态更新函数作为props传递给子组件,以及更推荐的、通过传递特定操作回调函数实现父组件数据过滤的两种模式,旨在提升组件间数据流的清…

    2025年12月21日
    000
  • 确保暗色模式切换图标在页面重载后状态持久化的教程

    本教程旨在解决暗色模式切换图标在页面重载后状态不持久的问题。通过优化css样式以响应`html`元素的`darkmode`类,并引入javascript初始化逻辑,确保图标状态与`localstorage`中存储的暗色模式设置同步,从而在页面加载时正确显示对应的月亮或太阳图标。 引言:暗色模式状态持…

    2025年12月21日
    000
  • Node.js Express 应用中静态文件权限问题的解决指南

    本文旨在解决node.js express应用在提供静态文件时常见的eacces: permission denied错误。通过深入分析文件系统权限机制,特别是当应用尝试访问非应用目录下的资源时,详细阐述了如何通过创建专用系统用户、正确配置文件和目录所有权,以及以受限用户身份运行应用来确保安全且可靠…

    2025年12月21日
    000
  • Intro.js教程:在引导消息中集成富文本与自定义HTML元素

    intro.js不仅支持纯文本引导消息,其`intro`属性还允许直接嵌入完整的html内容。这使得开发者能够在引导步骤中集成富文本、自定义ui元素乃至交互式组件,极大地增强了用户引导的灵活性和表现力,为用户提供更丰富、更具吸引力的引导体验。 在Intro.js引导消息中嵌入自定义HTML元素 In…

    2025年12月21日
    000
  • javascript的Node.js是什么_如何用js编写服务器端代码?

    Node.js 是一个基于 Chrome V8 引擎的 JavaScript 运行时环境,使 JS 能在服务器端运行;它非语言也非框架,而是提供 fs、http 等 API 的执行平台,核心为单线程+事件驱动+非阻塞 I/O,支持统一语言栈与庞大 npm 生态。 Node.js 是一个基于 Chro…

    2025年12月21日
    000
  • 使用JavaScript动态管理HTML元素类名:自动化移除与持久化修改

    本文详细介绍了如何使用javascript动态且自动化地移除html元素的特定css类名,以解决页面刷新后类名重新出现的问题。通过利用`document.queryselectorall`选择目标元素和`classlist.remove`方法,开发者可以有效地解除元素的功能限制(如`read-onl…

    2025年12月21日
    000
  • JavaScript同步控制轮播组件:解决文本内容切换与动画联动问题

    本教程旨在解决使用javascript同步控制轮播组件时,文本内容切换与视觉动画不同步的问题。通过分析代码中常见的变量作用域陷阱,特别是全局变量与局部变量的正确使用,我们将展示如何确保轮播的文本描述能够与旋转的视觉元素无缝联动,实现一个功能完善且逻辑清晰的多项轮播效果。 引言:同步轮播组件的需求与挑…

    2025年12月21日
    000
  • Playwright无障碍性测试实践:从DOM到可访问性树的探索与现代工具应用

    本文探讨了使用playwright进行无障碍性测试时,如何有效获取和分析页面的可访问性树(accessibility tree, at)。针对`page.accessibility.snapshot()`方法的局限性及其已弃用状态,文章重点推荐并演示了如何集成和使用业界标准的`@axe-core/p…

    2025年12月21日
    000
  • React Navigation中屏幕间参数传递的常见陷阱与解决方案

    本教程旨在解决React Native应用中使用React Navigation进行屏幕间参数传递时遇到的`undefined`错误。文章将深入分析参数传递的机制,揭示导致此类问题的常见原因,并提供一个具体的代码示例,展示如何正确地从`route.params`中解构和访问嵌套或独立传递的参数,确保…

    2025年12月21日
    000
  • WebGL纹理单元限制与跨平台优化策略

    本文深入探讨了WebGL中`MAX_COMBINED_TEXTURE_IMAGE_UNITS`等纹理单元参数在不同浏览器和环境下的差异性,并指出这些限制由硬件、驱动及底层API决定,无法通过编程强制提升。文章强调,与其追求高数值,不如通过高效的数据打包和纹理管理策略(如纹理图集、数据通道复用)来优化…

    2025年12月21日
    000
  • BPMN.js:实现序列流条件与名称的联动更新

    本文详细阐述了如何在bpmn-js中,通过监听模型变化并利用建模服务,实现序列流的名称(标签)与其条件表达式内容自动同步更新。文章将提供具体的代码示例,指导开发者正确处理事件拦截、属性更新及确保图形界面同步渲染的关键步骤。 概述 在BPMN模型设计中,序列流(Sequence Flow)的条件表达式…

    2025年12月21日
    000
  • JavaScript 热键优化日期输入控件:解决跨年日期计算问题

    本教程详细介绍了如何为日期输入控件实现高效的javascript热键功能,包括快速输入今日日期、按月、季度、年份增减以及按天增减。文章重点分析并解决了在进行日期计算时常见的跨年逻辑错误,通过优化javascript date对象的初始化方式,确保日期操作的准确性和鲁棒性,从而显著提升用户输入效率。 …

    2025年12月21日
    000
  • Next.js应用中基于版本控制的LocalStorage自动清理策略

    本文旨在解决next.js应用更新后,用户需手动清理localstorage和缓存以获取最新功能的问题。我们将介绍一种高效的解决方案,通过在客户端实现版本号比对机制,自动检测应用版本更新并清除旧的localstorage数据,确保用户始终使用最新版本的应用状态,从而优化用户体验并简化维护流程。 引言…

    2025年12月21日
    000

发表回复

登录后才能评论
关注微信