JavaScript设计模式之单例模式原理与用法实例分析


Posted in Javascript onJuly 26, 2018

本文实例讲述了JavaScript设计模式之单例模式原理与用法。分享给大家供大家参考,具体如下:

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

单例模式是一种常用的模式,有些对象只需要一个,如线程池、全局缓存、浏览器中的window对象等,这时候可以用到单例模式。

单例模式典型的应用场景:单击按钮时,页面中会出现一个登陆浮窗,而该登录浮窗是唯一的,无论单击多少次按钮,这个浮窗都会被创建一次,则适合用单例模式创建。

全局变量不是单例模式,但在JavaScript开发中,经常会把全局变量当成单例来使用。

使用var a = {};这种方式创建对象a时,对象a是独一无二的,若变量a被声明在全局作用域下,则可以在代码的任何位置使用这个变量。这显然满足单例模式的两个条件。

但是全局变量存在很多问题,很容易造成命名空间污染,如上面的var a = {};随时有可能被覆盖。

有必要尽量减少全局变量的使用,即使需要,也应该把它的污染降到最低。

降低全局变量带来的命名污染的几种方式:

1) 使用命名空间

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

把a和b都定义为namespace的属性,这样可以减少变量和全局作用域打交道的机会:

var namespace = {
 a: function() {
  alert("a");
 },
 b: function() {
  alert("b");
 }
};

动态地创建命名空间:

var obj = {};
obj.namespace = function(name) {
 var tips = name.split('.');
 var cur = obj;
 for (var i in tips) {
  if (!cur[tips[i]])
   cur[tips[i]] = {};
  cur = cur[tips[i]];
 }
};
obj.namespace('name');
obj.namespace('birth.year');
console.dir(obj);

上述代码等价于:

var obj = {
 name: {},
 birth: {
  year: {}
 }
};

2) 使用闭包封装私有变量

var person = (function() {
 var_name = "Alice";
 var _id = 16;
 return {
  getUserInfo: function() {
   return _name + ": " + _id;
  }
 }
})();

使用下划线来约定私有变量_name_age,它们被封装在闭包产生的作用域中,外部是访问不到这两个变量的,这就避免了对全局的命令污染。

惰性单例模式:

在需要的时候才创建对象实例。

var createBox = (function() {
 var div;
 return function() {
  if (!div) {
   div = document.createElement('div');
   div.innerHTML = '登录';
   div.style.display = 'none';
   document.body.appendChild(div);
  }
  return div;
 }
})();
document.getElementById('btn').onclick = function() {
 var box = createBox();
 box.style.display = 'block';
};

用变量div来判断是否已经创建过浮窗。

通用的惰性单例:

问题:上面的惰性单例实例是违反单一职责原则的,创建对象和管理单例的逻辑都放在createBox对象内部。若下次要创建页面中唯一的iframe,需要把createBox几乎照抄一遍。

var createIframe = (function() {
 var iframe;
 return function() {
  if (!iframe) {
   iframe = document.createElement('iframe');
   document.body.appendChild(iframe);
   return iframe;
  }
  return div;
 }
})();

解决:把不变的部分隔离出来,其实,管理单例的逻辑可以完全抽象出来,因为它们的逻辑是一样的:用一个变量来标记是否创建过对象,若是,则在下次直接返回已经创建好的对象。

var createSingle = function(func) {
 var flag;
 return flag || (flag = func.apply(this, arguments));
};
var createBox = function() {
 var div = document.createElement('div');
 div.innerHTML = '登录';
 div.style.display = 'none';
 document.body.appendChild(div);
 return div;
};
document.getElementById('btn').onclick = function() {
 var box = createBox();
 box.style.display = 'block';
};
var createIframe = createSingle(function() {
 var iframe = document.createElement('iframe');
 document.body.appendChild(iframe);
 return iframe;
});
document.getElementById('btn').onclick = function() {
 var iframe = createIframe();
 iframe.style.display = 'block';
};

单例模式的其他用途:

单例模式的用途远不止于创建对象,比如click事件只需要在第一次渲染页面时绑定一次,显然运用jQuery的one()方法可以实现。若运用createSingle 方法,也很容易实现:

var createSingle = function(func) {
 var flag;
 return flag || (flag = func.apply(this, arguments));
};
var bindEvent = createSingle(function() {
 document.getElementById(‘div').onclick = function() {
  ...
 };
 return true;
});
var render = function() {
 bindEvent();
};
render();
render();
render();

render()函数与bindEvent()函数执行了3次,但div实际上只被绑定了一次。

更多关于JavaScript相关内容可查看本站专题:《javascript面向对象入门教程》、《JavaScript切换特效与技巧总结》、《JavaScript查找算法技巧总结》、《JavaScript错误与调试技巧总结》、《JavaScript数据结构与算法技巧总结》、《JavaScript遍历算法与技巧总结》及《JavaScript数学运算用法总结》

希望本文所述对大家JavaScript程序设计有所帮助。

Javascript 相关文章推荐
jQuery中after的两种用法实例
Jul 03 Javascript
js时间比较示例分享(日期比较)
Mar 05 Javascript
jquery获取颜色在ie和ff下的区别示例介绍
Mar 28 Javascript
Angular Js文件上传之form-data
Aug 28 Javascript
jQuery中的基本选择器用法学习教程
Apr 14 Javascript
原生js实现放大镜效果
Jan 11 Javascript
JS实现unicode和UTF-8之间的互相转换互转
Jul 05 Javascript
深入理解Vue router的部分高级用法
Aug 15 Javascript
JavaScript实现抖音罗盘时钟
Oct 11 Javascript
JS实现秒杀倒计时特效
Jan 02 Javascript
Vue过渡效果之CSS过渡详解(结合transition,animation,animate.css)
Feb 05 Javascript
Element Badge标记的使用方法
Jul 27 Javascript
JavaScript设计模式之原型模式分析【ES5与ES6】
Jul 26 #Javascript
如何把vuejs打包出来的文件整合到springboot里
Jul 26 #Javascript
JavaScript中变量提升与函数提升经典实例分析
Jul 26 #Javascript
基于Vue实现微信小程序的图文编辑器
Jul 25 #Javascript
详解ECMAScript typeof用法
Jul 25 #Javascript
微信小程序动态生成二维码的实现代码
Jul 25 #Javascript
JavaScript设计模式之装饰者模式定义与应用示例
Jul 25 #Javascript
You might like
PHP 通过Socket收发十六进制数据的实现代码
2013/08/16 PHP
php+ajax无刷新上传图片实例代码
2015/11/17 PHP
PHP实现的微信公众号扫码模拟登录功能示例
2019/05/30 PHP
PHP 构造函数和析构函数原理与用法分析
2020/04/21 PHP
用document.documentElement取代document.body的原因分析
2009/11/12 Javascript
js列举css中所有图标的实现代码
2011/07/04 Javascript
基于jQuery插件实现点击小图显示大图效果
2016/05/11 Javascript
详解vue-cli构建项目反向代理配置
2017/09/07 Javascript
vue中element-ui表格缩略图悬浮放大功能的实例代码
2018/06/26 Javascript
移动端手指操控左右滑动的菜单
2019/09/08 Javascript
JS操作json对象key、value的常用方法分析
2019/10/29 Javascript
[06:45]DOTA2卡尔工作室 英雄介绍幻影长矛手篇
2013/07/12 DOTA
python在windows下创建隐藏窗口子进程的方法
2015/06/04 Python
Django框架中的对象列表视图使用示例
2015/07/21 Python
PHP网页抓取之抓取百度贴吧邮箱数据代码分享
2016/04/13 Python
对Python实现简单的API接口实例讲解
2018/12/10 Python
对pandas读取中文unicode的csv和添加行标题的方法详解
2018/12/12 Python
Python3模拟登录操作实例分析
2019/03/12 Python
python中p-value的实现方式
2019/12/16 Python
Python并发请求下限制QPS(每秒查询率)的实现代码
2020/06/05 Python
Python 分布式缓存之Reids数据类型操作详解
2020/06/24 Python
python 利用opencv实现图像网络传输
2020/11/12 Python
使用Python快速打开一个百万行级别的超大Excel文件的方法
2021/03/02 Python
CSS去掉A标签(链接)虚线框的方法
2014/04/01 HTML / CSS
瑞典度假品牌:OAS
2019/05/28 全球购物
儿科护士自我鉴定
2013/10/14 职场文书
英语演讲稿范文
2014/01/03 职场文书
好人好事事迹材料
2014/02/12 职场文书
购房个人委托书范本
2014/10/11 职场文书
全国法制宣传日活动总结2014
2014/11/01 职场文书
2015年学校办公室主任工作总结
2015/07/20 职场文书
2015年机关作风和效能建设工作总结
2015/07/23 职场文书
小学体育跳绳课教学反思
2016/02/16 职场文书
nginx location中多个if里面proxy_pass的方法
2021/03/31 Servers
python_tkinter事件类型详情
2022/03/20 Python
MySQL数据库Innodb 引擎实现mvcc锁
2022/05/06 MySQL