单例模式

July 25, 2019

单例模式「Singleton Pattern」:指一个类只有一个实例,且该类能自行创建这个实例的一种模式,为了节省内存资源、保证数据内容的一致性。

模式动机

在某些系统中,只有一个实例是非常必要的。比如任务管理器,首先从需求上来讲,一个任务管理器窗口完全满足需要。多个任务管理器窗口只会造成额外的开销,还需要处理各个窗口数据同步的问题。

实现

用一个变量来标记当前的类是否创建过对象,如果没创建过,返回创建之后的实例,如果之前已经创建过,则返回之前创建过的实例

单例模式「任务管理器」

代码

function TaskManage(name) {
  this.name
}

TaskManage.getInstance = (function() {
  let instance = null
  return function(name) {
    if (!instance) {
      instance = new TaskManage(name)
    }
    return instance
  }
})()

// 获取对象1

const a = TaskManage.getInstance('a')
// 获取对象2
const b = TaskManage.getInstance('b')
// 进行比较
console.info(a === b) // true

可以看到 ab 是相等的,但是假如我们想让打开的英雄联盟也是单例,我们就必须再写一个英雄联盟的类,并添加 getInstance 方法,这样显然 getInstance 是重复编写的,我们可以将其抽出来单独作为一个类。

通用的单例中间类

首先我们将控制生成单例类的 getInstance 方法抽离出来

// 获取单独的实例
const singleton = function(fn) {
    const instance;
    return function() {
        return instance || (instance = fn.apply(this, arguments));
    }
}

// 个人认为这样也可以
const singleton = function(Fn) {
    const instance
    return function() {
        return instance || (instance = new Fn(...arguments))
    }
}

我们再编写一个任务管理器的类

const taskManage = function(name) {
  this.name = name
}

taskManage.prototype.getName = function() {
  return this.name
}

然后我们将 taskManage 变为单例

const TaskManageSingleton = singleton(taskManage)
const win = TaskManageSingleton('win')
const mac = TaskManageSingleton('mac')

console.info(win.getName()) // 'win'
console.info(mac.getName()) // 'win'

可以看到第二次调用 TaskManageSingleton 时传入的参数并未生效,可以得知返回的是之前已经生成的 TaskManage 实例。

小结

单例模式确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例,这个类称为单例类,它提供全局访问的方法。单例模式的要点有三个:一是某个类只能有一个实例;二是它必须自行创建这个实例;三是它必须自行向整个系统提供这个实例。单例模式是一种对象创建型模式。