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 (面向切面编程)。这样做的价值在于把函数指责分开了,单例的工厂函数只负责缓存责任,实现具体业务的函数则不去负责这个事情,遵守函数单一职责原则。

相关推荐
waterHBO几秒前
css 模拟一个动画效果,消息堆叠。
前端·css
前端加油站16 分钟前
在 cursor 成为流行的开发方式后,作为普通开发我们能做什么
前端
Cache技术分享24 分钟前
163. Java Lambda 表达式 - Function 的链式组合
前端·后端
Zestia41 分钟前
页面点击跳转源代码?——element-jumper插件实现
前端·javascript
前端小白199541 分钟前
面试取经:工程化篇-webpack性能优化之优化loader性能
前端·面试·前端工程化
PineappleCoder41 分钟前
大小写 + 标点全搞定!JS 如何精准统计单词频率?
前端·javascript·算法
zhangbao90s43 分钟前
Web组件:使用Shadow DOM
前端
hhy前端之旅43 分钟前
语义版本控制:掌握版本管理的艺术
前端
coding随想43 分钟前
深入浅出DOM操作的隐藏利器:Range(范围)对象——掌控文档的“手术刀”
前端
前端小白199544 分钟前
面试取经:工程化篇-webpack性能优化之减少模块解析
前端·面试·前端工程化