js 单例模式 6 种实现方式

JavaScript 中的单例模式确保一个类只有一个实例,并提供全局访问点。以下是几种常见的实现方式:

1.对象字面量(最简单的方式)

javascript 复制代码
const Singleton = {
  property: 'value',
  method() {
    // 使用 this(在方法被解构调用时会丢失上下文)
    // console.log(this.property);
    
    // 使用 Singleton(更安全)
    console.log(Singleton.property);
  }
};

// 使用
Singleton.method();

JavaScript 可以使用对象字面量快速创建一个对象,创建的对象本身就是单例。这种方式最为简单,但是需要注意 this 的引用可能出问题。

为了解决 this 的问题,可以直接引用单例对象本身。

2.闭包实现

javascript 复制代码
const Singleton = (function() {
  let instance;
  
  function createInstance() {
    const object = new Object('I am the instance');
    return object;
  }
  
  return {
    getInstance: function() {
      if (!instance) {
        instance = createInstance();
      }
      return instance;
    }
  };
})();

// 使用
const instance1 = Singleton.getInstance();

这里利用了闭包的特性实现了模块的封装和单例对象的引用,返回一个 getInstance 方法用于获取实例对象。

3.ES6 Class 实现

javascript 复制代码
class Singleton {
  constructor() {
    if (Singleton.instance) {
      return Singleton.instance;
    }
    
    this.data = 'Singleton Data';
    Singleton.instance = this;
  }
  
  getData() {
    return this.data;
  }
  
  setData(data) {
    this.data = data;
  }
}

// 使用
const s1 = new Singleton();
const s2 = new Singleton();
console.log(s1 === s2); // true
s1.setData('New Data');
console.log(s2.getData()); // 'New Data'

这里的 instance 是一个静态变量,这种方式在 constructor 的最后将 this 赋值给 instance

4.改进的 class 实现(typescript 版本)

typescript 复制代码
class Singleton {
  private static instance?: Singleton;

  private constructor() {}

  static getInstance() {
    if (!Singleton.instance) {
      Singleton.instance = new Singleton();
    }
    return Singleton.instance;
  }
}

// 使用
const s1 = Singleton.getInstance();
const s2 = Singleton.getInstance();
console.log(s1 === s2); // true

typescript 版本抽取 getInstance 方法,让代码更可读。使用 private 关键字实现私有属性和方法,保证代码不会被随意篡改。

5.ES6 模块模式的单例

javascript 复制代码
// 在模块文件中
let instance = null;

class Database {
  constructor(config) {
    if (instance) {
      return instance;
    }
    
    this.connection = this.connect(config);
    instance = this;
  }
  
  connect(config) {
    return { connected: true, config };
  }
}

// 导出一个获取实例的函数
export const getInstance= (() => {
  let instance = null;
  return (config) => {
    if (!instance) {
      instance = new Database(config);
    }
    return instance;
  };
})();

这种方式利用 ES6 模块变量的引用共享特性,保证 instance 唯一。导出一个 getInstance 方法。

6.ES6 模块本身就是单例

javascript 复制代码
class Database {
  constructor(config) {
    this.connection = this.connect(config);
  }
  
  connect(config) {
    return { connected: true, config };
  }
}

// 直接导出一个实例
export default new Database({ host: 'localhost' });

实测证明,导出的实例在多个模块间是共享的。

总结

JavaScript 实现单例模式的方式很多,这里介绍常用的6种,主要分4大类:对象字面量、闭包实现、 ES6 class 实现、ES6 模块模式实现。

选择哪种实现方式取决于具体需求,简单场景可以使用对象字面量,复杂场景建议使用 ES6 Class 或 ES6 模块模式实现。

相关推荐
ldybk16 小时前
教学vue
前端·javascript·vue.js
卷卷说风控16 小时前
工作流的 Skill 怎么写?
java·javascript·人工智能·chrome·安全
还是大剑师兰特16 小时前
Pinia介绍及Vue3配置示例
前端·javascript·vue.js
还是大剑师兰特16 小时前
Vue3 Mixin 与 Vue2 Mixin 核心区别
前端·javascript·vue.js
188号安全攻城狮16 小时前
【前端基础知识】JavaScript 数组方法总结:从表格速查到分类详解
开发语言·前端·javascript·网络安全
英俊潇洒美少年16 小时前
迷你 React 调度器(带优先级+时间切片)手写实现
前端·javascript·react.js
freeWayWalker17 小时前
Vue通用缩放容器
前端·javascript·vue.js
Hello--_--World17 小时前
VUE:逻辑复用
前端·javascript·vue.js
如来神掌十八式17 小时前
设计模式之装饰器模式
java·设计模式
叫我一声阿雷吧17 小时前
JS 入门通关手册(43):async/await 原理与异常处理(实战 + 面试,彻底搞懂)
javascript·异常处理·promise·前端面试·async/await·generator·异步编程