Skip to content
On this page

前言

大家好,我是 Lvzl,本文属于我的专栏——前端需要掌握的设计模式系列,上一篇前端要掌握的设计模式——工厂模式。本文主要内容是单例模式

什么是单例模式

从名字上就能看出一些端倪,单例 —— 独一无二的实例,没错,就是指一个类只有一个实例,在一个应用中访问到的都是这个实例,这样的模式就是单例模式

如何实现单例模式

提供获取实例的静态方法

通过在 class 上提供一个静态方法用于获取该 class 的实例。

js
class Single {
  constructor() {
    //...
  }
  // 实例方法,通过实例对象 调用
  say() {
    console.log('我是一个单例对象')
  }
  // 静态方法,通过 Single.getInstance 调用
  static getInstance() {
    if (!Single.instance) {
      Single.instance = new Single()
    }
    return Single.instance
  }
}

const s1 = Single.getInstance()
const s2 = Single.getInstance()

console.log(s1 === s2) // true

闭包实现

js
class Single {
  constructor() {
    //...
  }
  say() {
    console.log('我是一个单例对象')
  }
}

Single.getInstance = (() => {
  let instance
  return () => {
    if (!instance) {
      instance = new Single()
    }
    return instance
  }
})()

const s1 = Single.getInstance()
const s2 = Single.getInstance()

console.log(s1 === s2) // true

以上两种实现都是提供了一个获取实例的方法,但是在工作中,我们在没有仔细看别人代码的时候,可能不会按别人预期那样创建类的实例,比如我们大概率会去直接 new,那上面这两种并不能保证获取到的实例唯一。那该如何实现呢?

new 多次也没问题

js
let Single = (() => {
  let instance
  class SingleClass {
    constructor() {
      //...
    }
    say() {
      console.log('我是一个单例对象')
    }
  }
  return function() {
    if (!instance) {
      instance = new SingleClass()
    }
    return instance
  }
})()

const s1 = new Single()
const s2 = new Single()

console.log(s1 === s2)

常见的单例模式应用

应用描述
VueXVuex 使用单一状态树,用一个对象就包含了全部的应用层级状态。至此它便作为一个“唯一数据源 (SSOT)”而存在。这也意味着,每个应用将仅仅包含一个 store 实例。单一状态树让我们能够直接地定位任一特定的状态片段,在调试的过程中也能轻易地取得整个当前应用状态的快照。image.png
VueRouter虽然官网没有说,但是从安装的逻辑是可以看出来的:image.png
还有很多...

总结

  1. 什么是单例模式? 类的实例只有一个
  2. 如何实现单例模式? 类的静态方法 或 闭包实现
  3. 常见的应用有哪些? 在一个一个应用中全局唯一的实例,就可以用单例模式实现。