JS 设计模式之:单例模式定义与实现方法浅析

作者:savokiss 时间:2024-04-29 14:10:04 

本文实例讲述了JS 设计模式之:单例模式定义与实现方法。分享给大家供大家参考,具体如下:

良好的设计模式可以显著提高代码的可读性,降低复杂度和维护成本。笔者打算通过几篇文章通俗地讲一讲常见的或者实用的设计模式。

今天先从最简单的一个入手:单例模式。

文中的示例代码会使用 ES6 语法,尽量简化不必要的细节

概念

单例模式(Singleton)属于创建型的设计模式,它限制我们只能创建单一对象或者某个类的单一实例。

通常情况下,使用该模式是为了控制整个应用程序的状态。在日常的开发中,我们遇到的单例模式可能有:Vuex 中的 StoreVue 的根实例任何导出单个对象的 ES6 模块等。

字面量写法

最简单的单例其实就像下面这样:


const cat = {
 name: 'mi',
 age: 4
}

了解 const 语法的小伙伴都知道,这只喵是不能被重新赋值的,但是它里面的属性其实是可变的。

如果想要一个不可变的单例对象:


const cat = {
 name: 'mi',
 age: 4
}

Object.freeze(cat);

这样就不能新增或修改这只喵上的任何属性,它变成了 冰冻喵~

如果是在模块中使用,上面的写法并不会污染全局作用域,但是直接生成一个固定的对象缺少了一些灵活性。

常用写法

相对而言,使用类或工厂方法来实现单例更加常用。假设我们有一个叫作 Logger 的类,它具有和 Console 相同的 API。

类单例

类的单例写法非常常用,如果我们想要这么使用它:


const logger = new Logger();
logger.log('msg');

// 这里大概写了 1000 行代码

const logger2 = new Logger();
logger.log('new msg');

logger === logger2; // true

即尽管 new 了多次 Logger,它返回的都是同一个实例。

下面直接看最实用的实现方式:


class Logger {
 constructor () {
   if (!Logger._singleton) {
     Logger._singleton = this;
   }
   return Logger._singleton;
 }

log (...args) {
   console.log(...args);
 }
}

export default Logger;

上面的方式将单例对象存储在了构造器上,这样的话不管 new Logger 多少次,返回的都是同一个 Logger 实例了。

这里有一个细节需要注意,即 new 关键字后面的构造函数如果显式返回一个对象,new 表达式就会返回该对象。

具体可参见 《你不知道的 JavaScript (上卷)》中的 new 绑定 相关章节。

工厂单例

如果不喜欢用 new 关键字,可以使用工厂方法返回单例对象。


let logger = null

class Logger {
 log (...args) {
   console.log(...args);
 }
}

function createLogger() {
 if (!logger) {
   logger = new Logger();
 }
 return logger;
}

export default createLogger;

上面的代码相当于在模块内部缓存了 logger 实例,然后导出了一个工厂方法。这种写法在模块化代码中比较常见,工厂方法也可以接收参数用来初始化单例对象。

今天的内容比较好理解,其中的单例写法也是笔者常用的方法。

下一篇我们再具体讲讲工厂模式的应用~

参考内容

  • 《JavaScript 设计模式》

  • 《JavaScript 面向对象编程指南》

  • 《你不知道的 JavaScript (上卷)》

  • Working with Singletons in JavaScript

希望本文所述对大家JavaScript程序设计有所帮助。

来源:https://segmentfault.com/a/1190000021874853

标签:JS,设计模式,单例模式
0
投稿

猜你喜欢

  • PHP开发技巧之PHAR反序列化详解

    2023-11-15 02:23:45
  • 本机安装PaddlePaddle安装指南及步骤详解

    2023-12-18 19:35:57
  • python中from module import * 的一个坑

    2021-10-29 08:23:51
  • 详解Ubuntu Server下启动/停止/重启MySQL数据库的三种方式

    2024-01-26 12:42:11
  • Python实现连接MySQL数据库的常见方法总结

    2024-01-22 05:28:26
  • 使用Python的Django和layim实现即时通讯的方法

    2022-05-29 00:41:33
  • 一文教你利用Python画花样图

    2023-09-01 20:26:42
  • 修改MySQL的数据库引擎为INNODB的方法

    2024-01-13 08:13:43
  • jsonpath做接口封装使用技巧

    2024-04-18 09:52:01
  • JS求1到任意数之间的所有质数的方法详解

    2023-06-28 11:46:06
  • asp下用fso和ado.stream写xml文件的方法

    2011-04-07 10:55:00
  • vue 指令版富文本溢出省略截取示例详解

    2024-05-28 15:52:36
  • golang 中strings包的Replace的使用说明

    2024-02-09 16:02:04
  • 关于python中逆序的三位数

    2021-08-09 05:17:28
  • Python进程池Pool应用实例分析

    2022-02-22 16:46:01
  • asp如何用数据库制作一个多用户版的计数器?

    2010-06-16 09:51:00
  • 在JavaScript中使用inline函数的问题

    2024-04-29 13:42:44
  • 研究Python的ORM框架中的SQLAlchemy库的映射关系

    2021-05-23 14:31:07
  • python对gif图压缩的完美解决方案

    2021-06-19 03:09:00
  • Python使用QQ邮箱发送邮件报错smtplib.SMTPAuthenticationError

    2023-07-07 06:58:52
  • asp之家 网络编程 m.aspxhome.com