设计模式
单例模式
单例模式是指确保一个类在任何情况下都绝对只有一个实例,并提供一个全局访问点。单例模式是创建型模式。
饿汉式单例模式
在类加载的时候就马上初始化了,此时还没到运行时只是将打包的代码加载到内存的时候就初始化(也就是线程还没有出现以前这个时间段初始化单例对象),所以此时一定是线程安全的,也就不可能存在访问安全问题
class HungrySingleton {
private static instance: HungrySingleton = new HungrySingleton();
public a = 9;
private constructor() {}
public static getInstance(): HungrySingleton {
return HungrySingleton.instance;
}
public add(a: number) {
this.a += a;
}
//公共的方法
}
- 优缺点
- 优点:
- 线程安全
- 执行效率高(因为在运行的时根本不需要再去创建对象的
- 缺点
- 可能造成内容浪费。因为在程序运行过程中,把所有在运行的单例类都用上了,那肯定是没有内存浪费的,但凡有一个类没有用上,就会造成内存浪费,因为在运行前这个对象已经实例化了。
- 灵活性受限: 由于实例在类加载时就创建,无法延迟实例化,因此无法通过传递参数来定制实例。
- 可能引起启动慢: 如果单例的初始化操作较为复杂,
- 优点:
懒汉式单例模式
上文说到了饿汉式可能会造成内存浪费,为了解决这个问题,也就出现了懒汉式单例模式。懒汉式单例模式的特点是,单例对象要在被使用的时候才会初始化
class LazySingleton {
private static instance: LazySingleton | null = null;
public a = 9;
private constructor(args: number) {
this.a = args;
}
public static getInstance(args: number): LazySingleton {
if (LazySingleton.instance == null) {
LazySingleton.instance = new LazySingleton(args);
}
return LazySingleton.instance;
}
public add(a: number) {
this.a += a;
}
}
优点:
- 延迟加载: 只有在需要时才会创建实例,避免了在程序启动时就占用内存的问题,节省了资源。优点:
- 线程安全性高(当需要时才创建): 在多线程环境下,只有在第一次创建实例时才会存在竞争,后续调用
getInstance()
方法都会返回已经创建好的实例,保证了线程安全性。
缺点:
- 线程安全性低(当实例创建时): 在多线程环境下,当多个线程同时调用
getInstance()
方法且实例尚未创建时,可能会导致多个实例被创建,破坏了单例模式的初衷。 - 可能存在性能问题: 在多线程环境下,每次调用
getInstance()
方法都需要进行同步,可能会造成性能上的开销。
常见使用场景
- 资源共享: 当应用程序需要共享资源时,例如数据库连接池、线程池、日志管理器等,单例模式可以确保所有对象共享相同的实例,避免资源浪费和不一致性。
- 配置管理器: 在整个应用程序中,可能有许多类需要访问相同的配置信息,例如数据库连接参数、日志级别等。单例模式可以用于创建一个全局的配置管理器,确保所有类都使用相同的配置实例。
- 日志记录器: 日志记录器是许多应用程序中常见的组件。使用单例模式可以确保在整个应用程序中只有一个日志记录器实例,从而简化日志记录的管理。
- 线程池管理器: 在多线程应用程序中,线程池管理器可以用于管理线程的生命周期、调度任务等。单例模式可以确保只有一个线程池管理器实例,避免资源竞争和不一致性。
- 缓存管理器: 在应用程序中使用缓存可以提高性能和降低系统负载。单例模式可以用于创建一个全局的缓存管理器,确保所有类都使用相同的缓存实例。
常用的单例模式库
TypeDI: TypeDI 是一个基于 TypeScript 的依赖注入(DI)库,它提供了单例和作用域绑定等功能。虽然主要用于依赖注入,但你可以使用 TypeDI 来方便地定义和管理单例对象。
InversifyJS: InversifyJS 是另一个流行的 TypeScript 依赖注入库,它支持单例、瞬时和作用域绑定等生命周期管理。类似于 TypeDI,你可以使用 InversifyJS 来定义和管理单例对象。
tsyringe: tsyringe 是一个轻量级的依赖注入库,适用于 TypeScript 和 JavaScript。它支持单例、瞬时和作用域绑定等生命周期管理。虽然主要用于依赖注入,但你也可以使用 tsyringe 来创建单例对象。