JavaScript设计模式(二)单例模式

单例模式指的是无论调用多少次生成函数,都只会返回相同的实例。在这之前,你需要对闭包有些许了解.

假设我们需要创建一个可复用的弹窗,一旦有了这个弹窗,就不再重复创建 div,大概逻辑如下:

js 复制代码
let temp = null; // 单例模式的关键变量
function Modal() {
  // 如果有缓存 直接返回缓存内容
  if (temp) {
    return temp;
  }
  return (temp = document.createElement("div"));
}

上面的代码非常的简单,仅仅是使用一个 temp 变量缓存了 div 这个节点,一旦创建过就不会重复创建,这就是单例模式的核心原理:缓存。接下来让我们完善这个功能:

js 复制代码
let temp = null; // 缓存变量 用于缓存稍后创建的弹窗
/**
 * 创建弹窗函数
 * @param {弹窗内容} modalContent
 */
function Modal(modalContent) {
  // 如果有缓存 直接返回缓存内容
  if (temp) {
    temp.innerHTML = modalContent;
    return temp;
  }
  const body = document.querySelector("body"); // 如果没有创建新的弹窗
  const div = document.createElement("div");
  div.innerHTML = modalContent;
  body.appendChild(div);
  return (temp = div);
}

现在我们来使用这个弹窗函数:

js 复制代码
const modal1 = Modal("modal1"); // 第一次执行由于没有缓存,会执行后面的创建过程
const modal2 = Modal("modal2"); // 第二次有了缓存,直接返回了temp 正是第一次创建好的temp
modal1 === modal12; // 结果是 true

到此为止单例模式的核心已经讲完了,有项目经验的同学肯定发现了 temp 是一个外部变量,会污染命名空间,我们需要使用闭包的方式创建缓存变量,闭包可以简单解释为是函数创建的变量环境,属于函数上下文,作用是不会污染全局上下文。

现在我们对上述例子进行改造: 其实除了使用 IIFE 立即执行函数创建闭包变量以外,没有什么其他的不同,但是这个例子仍不是完美的,看完示例代码你可以自己思考一分钟,答案会在后面揭晓。

js 复制代码
var Modal = (function (modalContent) {
  let temp = null; // 缓存变量 用于缓存稍后创建的弹窗
  return function () {
    // 如果有缓存 直接返回缓存内容
    if (temp) {
      temp.innerHTML = modalContent;
      return temp;
    }
    const body = document.querySelector("body"); // 如果没有创建新的弹窗
    const div = document.createElement("div");
    div.innerHTML = modalContent;
    body.appendChild(div);
    return (temp = div);
  };
})();

上述例子到底有什么缺陷呢,假如我今天需要复用弹窗,明天需要复用其他的东西,就不得不再次为某个具体的业务创造单例模式,那么我们在单例模式使用到达一定数量的时候,就需要考虑使用工厂函数:

js 复制代码
/** 单例工厂函数 */
function SingletonbFactory(Fn) {
  let instance = null; // 实例缓存
  return function () {
    if (instance) return instance; // 如果有直接返回
    return (instance = new Fn(arguments));
  };
}

function Modal(modalContent) {
  const body = document.querySelector("body"); // 如果没有创建新的弹窗
  const div = document.createElement("div");
  div.innerHTML = modalContent;
  body.appendChild(div);
  return div;
}

const SingleModal = SingletonFactory(Modal); // 工厂函数包装需要作为单例的函数

const modal1 = SingleModal("modal1"); // 弹窗命名为 modal1
const modal2 = SingleModal("modal2"); // 弹窗命名为 modal2

modal1 === modal1; // true

可能你会好奇,为什么要这么设计,这不是增加了理解难度吗?是的,这会增加代码的复杂度,万事都是有两面性的,使用设计模式需要学习成本。但是这在大型项目中是非常有价值的,以来这种方式符合函数式编程的理念,专业术语叫做 AOP (面向切面编程)。这样做的价值在于把函数指责分开了,单例的工厂函数只负责缓存责任,实现具体业务的函数则不去负责这个事情,遵守函数单一职责原则。

相关推荐
hh随便起个名3 小时前
力扣二叉树的三种遍历
javascript·数据结构·算法·leetcode
我是小路路呀4 小时前
element级联选择器:已选中一个二级节点,随后又点击了一个一级节点(仅浏览,未确认选择),此时下拉框失去焦点并关闭
javascript·vue.js·elementui
程序员爱钓鱼4 小时前
Node.js 编程实战:文件读写操作
前端·后端·node.js
PineappleCoder4 小时前
工程化必备!SVG 雪碧图的最佳实践:ID 引用 + 缓存友好,无需手动算坐标
前端·性能优化
JIngJaneIL5 小时前
基于springboot + vue古城景区管理系统(源码+数据库+文档)
java·开发语言·前端·数据库·vue.js·spring boot·后端
敲敲了个代码5 小时前
隐式类型转换:哈基米 == 猫 ? true :false
开发语言·前端·javascript·学习·面试·web
澄江静如练_5 小时前
列表渲染(v-for)
前端·javascript·vue.js
JustHappy6 小时前
「chrome extensions🛠️」我写了一个超级简单的浏览器插件Vue开发模板
前端·javascript·github
Loo国昌6 小时前
Vue 3 前端工程化:架构、核心原理与生产实践
前端·vue.js·架构