前言
本文首先谈到单例模式,顾名思义,就是在整个系统中只有一个对象,且该对象应当是全局性的。在架构设计的过程,可以想想哪些内容是需要考虑一个全局唯一的,像系统的回收站,只能打开一个;再比如,日志应用,是都可以考虑使用单例模式?
单例模式 Singleton
- 一个类只有一个实例,且该类能自行创建
- 全局性:单例类对外提供一个访问该单例的全局访问点
应用场景
- 场景1:.需要生成唯一序列的环境,如:统计某网站的访问量
- 场景2:需要频繁实例化然后销毁的对象。
- 场景3:创建对象时耗时过多或者耗资源过多,但又经常用到的对象。
- 场景4:方便资源相互通信的环境
类结构图
实现方式
饿汉模式(立即加载)
- 当使用单例类时就会实例化出对应的对象,且会一直占用内存,即便你在后续没有使用它
懒汉模式(延迟加载)
- 当调用get方法时才会创建该实例
代码实现:
//【立即加载】 设计模式01 - 单例模式
export class Singleton {
// 私有化构造函数,防止外部调用
private constructor(){}
private static instance = new Singleton(); // 使用Singleton时就实例化一个对象
public static getInstance(){
return this.instance;
}
}
//【延迟加载】 设计模式01 - 单例模式
export class Singleton2 {
private constructor(){}
private static instance : Singleton2;
public static getInstance() {
if(this.instance == null) {
this.instance = new Singleton2(); // 在调用getInstancce方法时才实例化一个对象
}
return this.instance;
}
}
从上面的代码上可以看到
- 立即加载模式,模式简单,也避免了多线程同步问题,但是如果这种方式创建的实例从未使用,就会造成资源浪费
- 延迟加载模式,实现了即用即实例方式,只有使用的时候才会创建实例。因为nodejs是单线程,所以可以不用考虑多线程情况,但是在多线程环境下,一个线程执行完
if(this.instance == null)
时,另一个线程也完成判断,就会产生2个实例。我们需要保证线程安全,延迟加载,效率较高。可以通过加锁或者其他方式来保证线程安全。