Javascript单例模式的介绍和实例


Posted in Javascript onOctober 08, 2016

前言

其实和其他编程语言一样,Javascript同样拥有着很多种设计模式,比如单例模式、代理模式、观察者模式等,熟练运用Javascript的设计模式可以使我们的代码逻辑更加清晰,并且更加易于维护和重构。

本文将会介绍Javascript中的单例模式,在传统开发工程师眼里,单例就是保证一个类只有一个实例,实现的方法一般是先判断实例存在与否,如果存在直接返回,如果不存在就创建了再返回,这就确保了一个类只有一个实例对象。在JavaScript里,单例作为一个命名空间提供者,从全局命名空间里提供一个唯一的访问点来访问该对象。下面来看看详细介绍。

单例模式概念

首先什么是单例模式?可以这样理解:单例模式旨在保证一个类仅有一个实例,并提供一个全局的访问点。

可能有人还是不太理解单例的概念,那么你可以想象生活中的一些例子。比如注册账号的时候,如果我们注册的账号已经存在,那么系统会提示我们“账号已经存在是否使用该账号登陆”,我们无法再次创建一个一模一样的账号,除非把原账号注销。这就是单例模式的生动体现。

类似的例子还有网页上的登陆弹框,无论我们点击多少次登陆按钮,界面上始终只会显示一个登陆弹框,无法再创建第二个。

本文就将以登陆弹框为例介绍单例模式的使用方式。

单例模式实例

1.demo展示

Javascript单例模式的介绍和实例

2.代码展示

构建单例模式弹框实例的代码每个人可能写的都不一样,但是目的都是一个:构建一个全局唯一并可访问的弹框。接下来我们一步一步来实现这个例子。

(1)获取DOM对象

var $ = function(id) {
 return typeof id === 'string' ? document.getElementById(id) : id; 
};

首先为了便于之后关于DOM的一些操作,我们这里利用函数式编程的原理将获取目标id的元素对象方法封装了一下,直接利用$(id)就可以获取。

(2)弹框构造函数

var Modal = function(id, html) {
 this.html = html;
 this.id = id;
 this.open = false;
};

这里我们声明了一个Modal作为弹框的构造函数,并且再其内部定义了公有属性html、id和open。html用来定义弹框内部的内容,id用来给弹框定义id名称,open用于判断弹框是否打开。

(3)open方法

Modal.prototype.create = function() {
 if (!this.open) {
 var modal = document.createElement('div');

 modal.innerHTML = this.html;
 modal.id = this.id;
 document.body.appendChild(modal);

 setTimeout(function() {
 modal.classList.add('show');
 }, 0);

 this.open = true;
 }
};

我们在Modal的原型链上定义了create方法,方法内部我们创建并向DOM中插入弹框,同时给弹框加上一个class为"show"的动画效果。这里简单介绍下classList:

classList是一个比className更便捷的操作元素class的属性,不过在兼容性方面不兼容IE10以下版本:

Javascript单例模式的介绍和实例
classList兼容性

其提供的操作class方法和jQuery的类似,主要有

     add(class1, class2, ...) 在元素中添加一个或多个类名,类似jQuery的addClass()

     remove(class1, class2, ...) 移除元素中一个或多个类名,类似jQuery的removeClass()

     contains(class) 判断指定的类名是否存在,类似jQuery的hasClass()

这里我们用到了add方法给Modal添加show类。

(4)close方法

Modal.prototype.delete = function() {
 if (this.open) {
 var modal = $(this.id);

 modal.classList.add('hide'); 
 setTimeout(function() {
 document.body.removeChild(modal);
 }, 200);

 this.open = false;
 }
};

定义了open方法后我们这里定义关闭弹框的方法,在其内部给弹框对象添加hide类动画效果,最后在页面上移除弹框对象。

(5)创建实例

var createIntance = (function() {
 var instance;
 return function() {
 return instance || (instance = new Modal('modal', '这是一个弹框'))
 }
})();

这是实现单例模式的重要部分,我们来分析下知识点:

     1、使用闭包封装了instance私有变量并返回一个函数

     2、利用 || 语法判断如果instance不存在则执行后者的实例化Modal方法,存在则直接返回instance,确保了只存在一个弹框实例

本实例的创建也可以理解为代理模式的一部分。

(6)按钮操作

var operate = {
 setModal: null,
 open: function() {
 this.setModal = createIntance();
 this.setModal.create();
 },
 delete: function() {
 this.setModal ? this.setModal.delete() : '';
 }
};

这里我们将按钮操作放在operate对象里,使得打开和关闭操作可以通过this获取实例setModal

(7)绑定事件

$('open').onclick = function() {
 operate.open();
};

$('delete').onclick = function() {
 operate.delete();
};

最后我们将打开和删除方法绑定到两个按钮上去,至此我们用单例模式实现的弹框demo就实现了。

总结

本文只演示了单例模式的一种实现方式,至于如何构建一个通用的单例模式就交给有兴趣的读者了。以上就是这篇文章的全部内容了,希望能对大家的学习或者工作能带来一定的帮助。

Javascript 相关文章推荐
[原创]网络复制内容时常用的正则+editplus
Nov 30 Javascript
JS中Iframe之间传值的方法
Mar 11 Javascript
JSON 数字排序多字段排序介绍
Sep 18 Javascript
JS对话框_JS模态对话框showModalDialog用法总结
Jan 11 Javascript
JS验证逗号隔开可以是中文字母数字
Apr 22 Javascript
Three.js学习之Lamber材质和Phong材质
Aug 04 Javascript
在vue中通过axios异步使用echarts的方法
Jan 13 Javascript
angularJS1 url中携带参数的获取方法
Oct 09 Javascript
微信小程序下拉框功能的实例代码
Nov 06 Javascript
JavaScript刷新页面的几种方法总结
Mar 28 Javascript
微信小程序实现的一键连接wifi功能示例
Apr 24 Javascript
如何在postman测试用例中实现断言过程解析
Jul 09 Javascript
jquery把int类型转换成字符串类型的方法
Oct 07 #Javascript
jquery判断类型是不是number类型的实例代码
Oct 07 #Javascript
js判断是否为空和typeof的用法(详解)
Oct 07 #Javascript
浅谈js的ajax的异步和同步请求的问题
Oct 07 #Javascript
jquery删除table当前行的实例代码
Oct 07 #Javascript
JS控制FileUpload的上传文件类型实例代码
Oct 07 #Javascript
jQuery+正则+文本框只能输入数字的实现方法
Oct 07 #Javascript
You might like
php调用mysql数据 dbclass类
2011/05/07 PHP
PHP借助phpmailer发送邮件
2015/05/11 PHP
织梦sitemap地图实时推送给百度的教程
2015/08/03 PHP
thinkphp自定义权限管理之名称判断方法
2017/04/01 PHP
针对thinkPHP5框架存储过程bug重写的存储过程扩展类完整实例
2018/06/16 PHP
FireFox JavaScript全局Event对象
2009/06/14 Javascript
js定义对象或数组直接量时各浏览器对多余逗号的处理(json)
2011/03/05 Javascript
jQuery 常见操作实现方式和常用函数方法总结
2011/05/06 Javascript
javascript检测是否联网的实现代码
2014/09/28 Javascript
js钢琴按钮波浪式图片排列效果代码分享
2015/08/26 Javascript
jQuery实现可关闭固定于底(顶)部的工具条菜单效果
2015/11/06 Javascript
jQuery自适应轮播图插件Swiper用法示例
2016/08/24 Javascript
BootStrap Table后台分页时前台删除最后一页所有数据refresh刷新后无数据问题
2016/12/28 Javascript
微信小程序 详解页面跳转与返回并回传数据
2017/02/13 Javascript
jQuery实现锚点向下平滑滚动特效示例
2017/08/29 jQuery
Angular2 http jsonp的实例详解
2017/08/31 Javascript
原生js实现拖拽功能基本思路详解
2018/04/18 Javascript
JavaScript求一组数的最小公倍数和最大公约数常用算法详解【面向对象,回归迭代和循环】
2018/05/07 Javascript
如何通过setTimeout理解JS运行机制详解
2019/03/23 Javascript
手把手教你 CKEDITOR 4 扩展插件制作
2019/06/18 Javascript
VUE+elementui组件在table-cell单元格中绘制微型echarts图
2020/04/20 Javascript
[37:29]完美世界DOTA2联赛PWL S2 LBZS vs Forest 第二场 11.19
2020/11/19 DOTA
python实现在无须过多援引的情况下创建字典的方法
2014/09/25 Python
Python中input和raw_input的一点区别
2014/10/21 Python
Python图形绘制操作之正弦曲线实现方法分析
2017/12/25 Python
python海龟绘图之画国旗实例代码
2020/11/11 Python
通过代码简单了解django model序列化作用
2020/11/12 Python
详解CSS3中border-image的使用
2015/07/18 HTML / CSS
html5中canvas学习笔记1-画板的尺寸与实际显示尺寸
2013/01/06 HTML / CSS
韩国休闲女装品牌网站:ANAIS
2016/08/24 全球购物
Kneipp克奈圃美国官网:德国百年精油配方的传承
2018/02/07 全球购物
荷兰音乐会和音乐剧门票订购网站:Topticketshop
2019/08/27 全球购物
儿科护理实习自我鉴定
2013/09/19 职场文书
村委会主任先进事迹
2014/01/15 职场文书
红领巾心向党广播稿
2014/01/19 职场文书
阿里云服务器Ubuntu 20.04上安装Odoo 15
2022/05/20 Servers