·vincent

js中的单例模式

js设计模式 单例

Javascript

本次分享的内容是设计模式中的单例模式。本文介绍了什么是单例模式,单例模式的两种实现方式和各种的优缺点,最后介绍了如何借助ES6新特性Proxy实现包装一个单例模式。

单例模式

单例模式就是保证一个类只有一个实例,而且自行实例化并向整个系统提供这个实例。

单例模式的优点:

  1. 由于单例模式只生成一个实例,所以能减少系统性能的开销;

  2. 避免对共享资源的多重占用导致的性能损耗,如日志文件,应用配置。

  3. 提供了对唯一实例的受控访问,加快对象访问速度,比如多线程线程池的设计,方便对池中的线程进行控制。

单例模式的特点:

  1. 保证一个类中只有一个实例;

  2. 在类中实例化自己;

  3. 向整个系统提供这个实例;

单例模式的实现

单例模式被大家熟知的有两种实现方式:饿汉式单例模式和懒汉式单例模式。

两种实现方式的核心要点相同就是构造函数私有化,然后将一个静态私有变量确立为唯一实例,外部通过静态方法访问这个唯一的实例。

-懒汉式单例模式

ts
1class Singleton1{
2    /*核-一个接收实例的静态成员*/
3    private static people:Singleton1;
4    private name:string;
5    /*核心-私有构造函数*/
6    private constructor(name:string) {
7        this.name = name;
8    }
9    /*核心-获取实例*/
10    public static getInstance():Singleton1{
11        if(Singleton1.people == null){
12            Singleton1.people = new Singleton1("懒汉式单例模式");
13        }
14        return Singleton1.people;
15    }
16    public say():void{
17        console.log(`${this.name}`);
18    }
19}
20/*测试代码*/
21let people = Singleton1.getInstance();
22people.say(); //懒汉式单例模式

懒汉式单例模式在第一次调用时才会实例化对象,之后不会再实例化,直接返回第一次创建的对象。该模式资源利用率高,而且有延时加载的优势。但是调用效率低。

-饿汉式单例模式

ts
1class Singleton2{
2    private static people:Singleton2 = new Singleton2('饿汉式单例模式');
3    private name:string;
4    private constructor(name) {
5        this.name = name;
6    }
7    public static getInstance():Singleton2{
8        return Singleton2.people;
9    }
10    public say():void{\
11        console.log(this.name);
12    }
13}
14/*测试代码*/
15let s = Singleton2.getInstance();
16s.say(); //饿汉式单例模式

饿汉式单例模式类加载时立即实例化,之后返回实例化的对象。该模式调用效率高。因为加载时立刻实例化,所以没有延时加载的优势,并且没有用到这个实例的话,就会白白实例化一个对象,浪费资源。

Proxy实现单例模式

Proxy
是ES6语法新特性,用于修改对象某些操作的默认行为,也可以理解为在目标对象之前架设一层拦截,所有的访问都必须先通过这层拦截,因此提供了一种机制,可以对于外部的访问进行过滤和修改。利用该特性,我们可以将一个非单例模式的类包装成为单例模式。

ts
1function singletonIfy(className):ProxyConstructor{
2    return new Proxy(className.prototype.constructor,{
3        construct(target: ProxyConstructor, argArray: any, newTarget?: any): object {
4            if(!this.instance){
5                // @ts-ignore
6                this.instance = new target(...argArray);
7            }
8            return this.instance;
9        }
10    })
11}
12
13class MyClass{
14    private readonly msg: string;
15    constructor(msg) {
16        this.msg = msg;
17    }
18    printMsg():void{
19        console.log(this.msg);
20    }
21}
22
23let MySingletonClass;
24MySingletonClass = singletonIfy(MyClass);
25const obj1 = new MySingletonClass('first');
26obj1.printMsg(); //first
27const obj2 = new MySingletonClass('second');
28obj2.printMsg(); //first

代理模式实现逻辑跟懒汉式单例模式相同,都是在调用时进行实例的初始化,之后的调用返回实例化的对象。

小结

本次分享使用TS介绍了单例模式的3种实现方式。

ts
1function singleClass (className){
2    let ins 
3    return new Proxy(className, {
4        construct(target, args) {
5            if(!ins){
6                ins = new target(...args)
7            }
8            return ins
9        }
10    })
11}
12class BVideo {
13    constructor(){
14        console.log('create')
15    }
16}
17const Video = singleClass(BVideo)
18Video.prototype.play = function(){
19    console.log('play')
20}
21const v1 = new Video()
22const v2 = new Video()
23v1.play()
24console.log('vvvvv', v1===v2)
ts
1export type Constructor<T> = new (...arg: any[]) => T;
2
3export function createSingleton<T extends object>(Cls: Constructor<T>): Constructor<T> {
4  let instance: T;
5  const handler: ProxyHandler<Constructor<T>> = {
6    construct(target, args: any[], newTarget: ProxyConstructor) {
7      if (!instance) {
8        instance = Reflect.construct(target, args, newTarget);
9      }
10      return instance;
11    },
12  };
13  return new Proxy(Cls, handler);
14}