Javascript单例模式概念与实例

VirBQH 8年前
   <h2><strong>前言</strong></h2>    <p>和其他编程语言一样,Javascript同样拥有着很多种设计模式,比如单例模式、代理模式、观察者模式等,熟练运用Javascript的设计模式可以使我们的代码逻辑更加清晰,并且更加易于维护和重构。</p>    <p>本文将介绍Javascript模式中较为常见和实用的模式——单例模式,主要分为概念和实例部分。在介绍实例的同时也会对代码中额外的知识点进行讲解。</p>    <h2><strong>单例模式概念</strong></h2>    <p>首先什么是单例模式?可以这样理解:单例模式旨在保证一个类仅有一个实例,并提供一个全局的访问点。</p>    <p>可能有人还是不太理解单例的概念,那么你可以想象生活中的一些例子。比如注册账号的时候,如果我们注册的账号已经存在,那么系统会提示我们“账号已经存在是否使用该账号登陆”,我们无法再次创建一个一模一样的账号,除非把原账号注销。这就是单例模式的生动体现。</p>    <p>类似的例子还有网页上的登陆弹框,无论我们点击多少次登陆按钮,界面上始终只会显示一个登陆弹框,无法再创建第二个。</p>    <p>本文就将以登陆弹框为例介绍单例模式的使用方式。</p>    <p>单例模式实例</p>    <p><strong>1.demo展示</strong></p>    <p style="text-align:center"><img src="https://simg.open-open.com/show/6994ea97609d52cbcb78de4317d07db3.gif"></p>    <p>演示地址为:弹框实例</p>    <p><strong>2.代码展示</strong></p>    <p>构建单例模式弹框实例的代码每个人可能写的都不一样,但是目的都是一个:构建一个全局唯一并可访问的弹框。接下来我们一步一步来实现这个例子。</p>    <p><strong>(1)获取DOM对象</strong></p>    <pre>  <code class="language-javascript">var $ = function(id) {      return typeof id === 'string' ? document.getElementById(id) : id;      };    </code></pre>    <p>首先为了便于之后关于DOM的一些操作,我们这里利用函数式编程的原理将获取目标id的元素对象方法封装了一下,直接利用$(id)就可以获取。</p>    <p><strong>(2)弹框构造函数</strong></p>    <pre>  <code class="language-javascript">var Modal = function(id, html) {      this.html = html;      this.id = id;      this.open = false;      };    </code></pre>    <p>这里我们声明了一个Modal作为弹框的构造函数,并且再其内部定义了公有属性html、id和open。html用来定义弹框内部的内容,id用来给弹框定义id名称,open用于判断弹框是否打开。</p>    <p><strong>(3)open方法</strong></p>    <pre>  <code class="language-javascript">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;       }   };    </code></pre>    <p>我们在Modal的原型链上定义了create方法,方法内部我们创建并向DOM中插入弹框,同时给弹框加上一个class为"show"的动画效果。这里简单介绍下classList:</p>    <p>classList是一个比className更便捷的操作元素class的属性,不过在兼容性方面不兼容IE10以下版本:</p>    <p style="text-align:center"><img src="https://simg.open-open.com/show/c676d64a2fb96e4e7f4963a03072e736.jpg"></p>    <p>其提供的操作class方法和jQuery的类似,主要有</p>    <ul>     <li>add(class1, class2, ...) 在元素中添加一个或多个类名,类似jQuery的addClass()</li>     <li>remove(class1, class2, ...) 移除元素中一个或多个类名,类似jQuery的removeClass()</li>     <li>contains(class) 判断指定的类名是否存在,类似jQuery的hasClass()</li>    </ul>    <p>这里我们用到了add方法给Modal添加show类。</p>    <p><strong>(4)close方法</strong></p>    <pre>  <code class="language-javascript">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;       }   };    </code></pre>    <p>定义了open方法后我们这里定义关闭弹框的方法,在其内部给弹框对象添加hide类动画效果,最后在页面上移除弹框对象。</p>    <p><strong>(5)创建实例</strong></p>    <pre>  <code class="language-javascript">var createIntance = (function() {       var instance;       return function() {           return instance || (instance = new Modal('modal', '这是一个弹框'))       }   })();    </code></pre>    <p>这是实现单例模式的重要部分,我们来分析下知识点:</p>    <ul>     <li>使用闭包封装了instance私有变量并返回一个函数</li>     <li>利用 || 语法判断如果instance不存在则执行后者的实例化Modal方法,存在则直接返回instance,确保了只存在一个弹框实例</li>    </ul>    <p>本实例的创建也可以理解为代理模式的一部分。</p>    <p><strong>(6)按钮操作</strong></p>    <pre>  <code class="language-javascript">var operate = {       setModal: null,       open: function() {           this.setModal = createIntance();           this.setModal.create();       },       delete: function() {           this.setModal ? this.setModal.delete() : '';       }   };   </code></pre>    <p>这里我们将按钮操作放在operate对象里,使得打开和关闭操作可以通过this获取实例setModal。</p>    <p><strong>(7)绑定事件</strong></p>    <pre>  <code class="language-javascript">$('open').onclick = function() {       operate.open();   };      $('delete').onclick = function() {       operate.delete();   };    </code></pre>    <p>最后我们将打开和删除方法绑定到两个按钮上去,至此我们用单例模式实现的弹框demo就实现了。</p>    <p> </p>    <p> </p>    <p>来自:http://developer.51cto.com/art/201610/518400.htm</p>    <p> </p>