js设计模式——单例模式

文章介绍了JavaScript中的单例模式,包括使用命名空间和闭包来减少全局变量,以及惰性单例的实现。在Vue中,Vuex作为全局状态管理器体现了单例模式。同时,提供了实现单例的Storage示例,一个是基于闭包,另一个是通过静态类方法。
摘要由CSDN通过智能技术生成

定义:保证一个类仅有一个实例,并提供一个访问它的全局访问点。

Vue 中对应的体现就是 Vuex,一个 Vue 实例只会有一个全局的 Store。

一、JavaScript 中的单例模式

在 JavaScript 开发中,我们经常会把全局变量当成单例来使用,但是这样容易造成命名空间污染。我们有必要尽量减少全局变量的使用,即使需要,也应该把污染降到最低。

1. 使用命名空间

适当的使用命名空间,并不会杜绝全局变量,但可以减少全局变量的数量。

let MyApp = {};

MyApp.namespace = (name) => {
  const parts = name.split('.')
  let current = MyApp
  for (var i in parts) {
    const key = parts[i]
    if ( !current[key] ) {
      current[key] = {}
    }
    current = current[key]
  }
};

MyApp.namespace('event')
MyApp.namespace('dom.stype')

console.dir(MyApp)

// 上述代码等价于

let MyApp = {
  event: {},
  dom: {
    style: {}
  }
};
2. 使用闭包封装私有变量

把一些变量封装到闭包内部,只暴露一些接口跟外界通信:

const user = (() => {
  const _name = 'sven', _age = 29
  return {
    getUserInfo: () => {
      return _name + '-' + _age
    }
  }
})();

console.log(user.getUserInfo())

二、惰性单例

惰性单例指的是在需要的时候才创建对象实例。

// 管理单例
const getSingle = function(fn) {
  let result
  return () => {
    return result || ( result = fn.apply(this, arguments) )
  }
}

const createLoginLayer = () => {
  const div = document.createElement('div')
  div.innerHTML = '我是登录浮窗'
  document.body.appendChild(div)
  return div
}

const createSingleLoginLayer = getSingle(createLoginLayer)
var loginLayer = createSingleLoginLayer()
var loginLayer = createSingleLoginLayer()
var loginLayer = createSingleLoginLayer()


const createSingleIframe = getSingle(() => {
  const iframe = document.createElement('iframe')
  document.body.appendChild(iframe)
  return iframe
})
var iframe = createSingleIframe()
var iframe = createSingleIframe()
var iframe = createSingleIframe()

三、例子

  • 描述(实现一个 Storage)
    实现 Storage,使得该对象为单例,基于 localStorage 进行封装。实现方法 setItem(key,value) 和 getItem(key)。

  • 实现 1 (闭包实现 Storage)

function StoreBase(){}
StoreBase.prototype.getItem = (key) => {
  return localStorage.getItem(key)
}
StoreBase.prototype.setItem = (key, val) => {
  localStorage.setItem(key, val)
}

const Store = (() => {
  let instance = null
  return () => {
    if (!instance) {
      instance = new StoreBase()
    }
    return instance
  }
})()

const store1 = Store()
const store2 = Store()

store1.setItem('name', '张三')

store2.getItem('name') // 张三
store1.getItem('name') // 张三

console.log(store1 === store2) //
  • 实现 2 (静态类实现 Storage)
class Storage {
  static getInstance() {
    if (!Storage.instance) {
      Storage.instance = new Storage()
    }
    return Storage.instance
  }
  getItem(key) {
    return localStorage.getItem(key)
  }
  setItem(key, value) {
    localStorage.setItem(key, value)
  }
}

const storage1 = Storage.getInstance()
const storage2 = Storage.getInstance()

storage1.setItem('name', '张三')
storage1.getItem('name')
storage2.getItem('name')

console.log(storage1 === storage2)
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值