前端单例对象实现

前言

最近在开发个人项目的过程中,涉及到一个场景需要创建一个单例对象(即只能实例化一次,后续无论如何实例化获取的都是最初实例化的对象)
但是js实现单例对象的时候始终有缺陷,即便是基于模块导出了一个已实例化完成的对象。也一样可以通过原型链的方式去获取单例对象的class。

因此在实现这个功能的过程中,我花了不少时间,下面讲一下我的单例对象的实现思路。

实现思路

在之前实现minivue3的过程中我频繁的使用到了proxy这个代理。
因此从中我获得灵感,我希望同样基于模块,我默认导出一个proxy对象,对象劫持的是目标class(class也是对象,js万物皆对象,构造函数只是其中一个熟悉),并在proxy对象中监听构造函数。
在实例化我们导出的proxy对象中,实际上会触发我们监听的构造函数。

如下所示:

在构造函数被触发的时候,可以返回已经实例化好的对象,这样无论怎么new 拿到的都是我们最初已经实例化完的对象。

代码实现

因此代码实现很简单

js 复制代码
// 封装proxy代理单例函数
function singleton(className) {
  let obj = '' // 通过闭包存储实例化的对象
  return new Proxy(className, {
    construct: function(target, args) {
      if(!obj) obj = new target(...args)
      return obj
    },
  });
}

使用时默认导出封装之后的单例函数即可,附上用例如下:

js 复制代码
// 目标对象
class student {
  constructor(name, age) {
    this.name = name;
    this.age = age;
  }
}

// 封装学生单例
let singletonStu = singleton(student)

// 测试用例
let c = new singletonStu('xiaoShan', 20)
let d = new singletonStu('jiajia', 19)
console.log(c === d)

效果如下:

总结

如果朋友们以后有遇到这种单例的情况不妨通过这种代理和模块结合的方式去实现,效果还是不错的。但需要留意proxy对象有可能在一些古早浏览器中不兼容,今天就分享到这里。

相关推荐
hedley(●'◡'●)11 分钟前
基于cesium和vue的大疆司空模仿程序
前端·javascript·vue.js·python·typescript·无人机
qq5_81151751513 分钟前
web城乡居民基本医疗信息管理系统信息管理系统源码-SpringBoot后端+Vue前端+MySQL【可直接运行】
前端·vue.js·spring boot
百思可瑞教育14 分钟前
构建自己的Vue UI组件库:从设计到发布
前端·javascript·vue.js·ui·百思可瑞教育·北京百思教育
百锦再14 分钟前
Vue高阶知识:利用 defineModel 特性开发搜索组件组合
前端·vue.js·学习·flutter·typescript·前端框架
CappuccinoRose39 分钟前
JavaScript 学习文档(二)
前端·javascript·学习·数据类型·运算符·箭头函数·变量声明
这儿有一堆花1 小时前
Vue 是什么:一套为「真实业务」而生的前端框架
前端·vue.js·前端框架
YigAin1 小时前
Unity23种设计模式之 享元模式
设计模式·享元模式
全栈前端老曹1 小时前
【MongoDB】深入研究副本集与高可用性——Replica Set 架构、故障转移、读写分离
前端·javascript·数据库·mongodb·架构·nosql·副本集
NCDS程序员1 小时前
v-model: /v-model/ :(v-bind)三者核心区别
前端·javascript·vue.js
夏幻灵2 小时前
CSS三大特性:层叠、继承与优先级解析
前端·css