解析Javascript单例模式概念与实例


Posted in Javascript onDecember 05, 2016

前言

和其他编程语言一样,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单例模式概念与实例

其提供的操作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', '这是一个弹框'))
 }
})();

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

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

利用 || 语法判断如果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就实现了。

结语

本文只演示了单例模式的一种实现方式,至于如何构建一个通用的单例模式就交给有兴趣的读者了。

文章灵感来源于《Javascrit设计模式与开发实践》一书。

以上就是本文的全部内容,希望对大家有所帮助,同时也希望多多支持三水点靠木!

Javascript 相关文章推荐
jQuery.Autocomplete实现自动完成功能(详解)
Jul 13 Javascript
6个DIV 135或246间隔一秒轮番显示效果
Jul 24 Javascript
利用google提供的API(JavaScript接口)获取网站访问者IP地理位置的代码详解
Jul 24 Javascript
克隆javascript对象的三个方法小结
Jan 12 Javascript
javascript 判断字符串是否包含某字符串及indexOf使用示例
Oct 18 Javascript
cocos2dx骨骼动画Armature源码剖析(一)
Sep 08 Javascript
jquery获取复选框checkbox的值实现方法
May 30 Javascript
JavaScript实现刷新不重记的倒计时
Aug 10 Javascript
javascript基础知识之html5轮播图实例讲解(44)
Feb 17 Javascript
JS中通过url动态获取图片大小的方法小结(两种方法)
Oct 31 Javascript
微信小程序自定义导航栏实例代码
Apr 05 Javascript
编写v-for循环的技巧汇总
Dec 01 Javascript
微信公众号  提示:Unauthorized API function 问题解决方法
Dec 05 #Javascript
深入理解jQuery()方法的构建原理
Dec 05 #Javascript
jQuery控制控件文本的长度的操作方法
Dec 05 #Javascript
浅析Ajax语法
Dec 05 #Javascript
jQuery的事件预绑定
Dec 05 #Javascript
微信小程序 UI布局常用技巧整理总结
Dec 05 #Javascript
深入理解vue.js双向绑定的实现原理
Dec 05 #Javascript
You might like
AM/FM收音机的安装与调试
2021/03/02 无线电
PHP测试程序运行时间的类
2012/02/05 PHP
php xml常用函数的集合(比较详细)
2013/06/06 PHP
修改apache配置文件去除thinkphp url中的index.php
2014/01/17 PHP
php curl_init函数用法
2014/01/31 PHP
PHPCrawl爬虫库实现抓取酷狗歌单的方法示例
2017/12/21 PHP
PHP使用函数用法详解
2018/09/30 PHP
js实现商品抛物线加入购物车特效
2020/11/18 Javascript
全面理解闭包机制
2016/07/11 Javascript
详解nodejs微信公众号开发——2.自动回复
2017/04/10 NodeJs
Vue.js实现在下拉列表区域外点击即可关闭下拉列表的功能(自定义下拉列表)
2017/05/30 Javascript
Vue.js图片预览插件使用详解
2018/08/27 Javascript
vue单页缓存存在的问题及解决方案(小结)
2018/09/25 Javascript
详解Vue.directive 自定义指令
2019/03/27 Javascript
vue3.0搭配.net core实现文件上传组件
2020/10/29 Javascript
[01:24]2014DOTA2 TI第二日 YYF表示这届谁赢都有可能
2014/07/11 DOTA
[01:02:38]DOTA2-DPC中国联赛定级赛 LBZS vs Phoenix BO3第二场 1月10日
2021/03/11 DOTA
pygame播放音乐的方法
2015/05/19 Python
Python 3.x读写csv文件中数字的方法示例
2017/08/29 Python
Python3实现的简单验证码识别功能示例
2018/05/02 Python
python 定时器,轮询定时器的实例
2019/02/20 Python
python+selenium+chromedriver实现爬虫示例代码
2020/04/10 Python
Python Tornado核心及相关原理详解
2020/06/24 Python
2020年10款优秀的Python第三方库,看看有你中意的吗?
2021/01/12 Python
python爬虫爬取某网站视频的示例代码
2021/02/20 Python
一款基于css3和jquery实现的动画显示弹出层按钮教程
2015/01/04 HTML / CSS
印尼在线旅游门户网站:NusaTrip
2019/11/01 全球购物
北体毕业生求职信
2014/02/28 职场文书
法律专业求职信
2014/05/24 职场文书
三月学雷锋活动总结
2014/06/26 职场文书
药店促销活动策划方案
2014/08/24 职场文书
留学推荐信(中英文版)
2015/03/26 职场文书
Python读取文件夹下的所有文件实例代码
2021/04/02 Python
CSS 文字装饰 text-decoration & text-emphasis 详解
2021/04/06 HTML / CSS
Springboot集成阿里云OSS上传文件系统教程
2021/06/28 Java/Android
漫画「日和酱的要求是绝对的」第3卷封面公开
2022/03/21 日漫