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 相关文章推荐
Js 中debug方式
Feb 07 Javascript
js显示时间 js显示最后修改时间
Jan 02 Javascript
Jquery UI震动效果实现原理及步骤
Feb 04 Javascript
文本域中换行符的替换示例
Mar 04 Javascript
Javascript控制input输入时间格式的方法
Jan 28 Javascript
JavaScript数据类型之基本类型和引用类型的值
Apr 01 Javascript
node网页分段渲染详解
Sep 05 Javascript
微信小程序  Mustache语法详细介绍
Oct 27 Javascript
JavaScript 轮播图和自定义滚动条配合鼠标滚轮分享代码贴
Oct 28 Javascript
深入理解vue2.0路由如何配置问题
Jul 18 Javascript
小程序云开发实战小结
Oct 25 Javascript
vue 解决兄弟组件、跨组件深层次的通信操作
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内置访问资源的超时时间 time_out file_get_contents read_file
2013/06/03 PHP
fsockopen pfsockopen函数被禁用,SMTP发送邮件不正常的解决方法
2015/09/20 PHP
thinkphp项目部署到Linux服务器上报错“模板不存在”如何解决
2016/04/27 PHP
[原创]php正则删除img标签的方法示例
2017/05/27 PHP
一个收集图片的bookmarlet(js 刷新页面中的图片)
2010/05/27 Javascript
JQuery的Validation插件中Remote验证的中文问题
2010/07/26 Javascript
JavaScript高级程序设计阅读笔记(六) ECMAScript中的运算符(二)
2012/02/27 Javascript
简单的JavaScript互斥锁分享
2014/02/02 Javascript
angularjs学习笔记之双向数据绑定
2015/09/26 Javascript
JS新包管理工具yarn和npm的对比与使用入门
2016/12/09 Javascript
概述jQuery中的ajax方法
2016/12/16 Javascript
web打印小结
2017/01/11 Javascript
node使用Koa2搭建web项目的方法
2017/10/17 Javascript
Angular 4中如何显示内容的CSS样式示例代码
2017/11/06 Javascript
使用vue-router设置每个页面的title方法
2018/02/11 Javascript
vue todo-list组件发布到npm上的方法
2018/04/04 Javascript
关于HTML5的data-*自定义属性的总结
2018/05/05 Javascript
教你完全理解ReentrantLock重入锁
2019/06/03 Javascript
python读取html中指定元素生成excle文件示例
2014/04/03 Python
python使用cPickle模块序列化实例
2014/09/25 Python
Python正则表达式非贪婪、多行匹配功能示例
2017/08/08 Python
利用Python实现在同一网络中的本地文件共享方法
2018/06/04 Python
浅谈Series和DataFrame中的sort_index方法
2018/06/07 Python
python基础学习之如何对元组各个元素进行命名详解
2018/07/12 Python
pyqt5 使用label控件实时显示时间的实例
2019/06/14 Python
Tensorflow 自定义loss的情况下初始化部分变量方式
2020/01/06 Python
解决python的空格和tab混淆而报错的问题
2021/02/26 Python
8款使用 CSS3 实现超炫的 Loading(加载)的动画效果
2015/03/17 HTML / CSS
使用canvas生成含有微信头像的邀请海报没有微信头像问题
2019/10/29 HTML / CSS
英国家用电器购物网站:Hughes
2018/02/23 全球购物
美国在线自行车商店:Jenson USA
2018/05/22 全球购物
巴西最大的在线约会网站:ParPerfeito
2018/07/11 全球购物
汽车制造与装配专业自荐信范文
2014/01/02 职场文书
给公司的建议书范文
2014/05/13 职场文书
写一个Python脚本自动爬取Bilibili小视频
2021/04/24 Python
如何利用js在两个html窗口间通信
2021/04/27 Javascript