深入理解JavaScript系列(37):设计模式之享元模式详解


Posted in Javascript onMarch 04, 2015

介绍

享元模式(Flyweight),运行共享技术有效地支持大量细粒度的对象,避免大量拥有相同内容的小类的开销(如耗费内存),使大家共享一个类(元类)。

享元模式可以避免大量非常相似类的开销,在程序设计中,有时需要生产大量细粒度的类实例来表示数据,如果能发现这些实例除了几个参数以外,开销基本相同的 话,就可以大幅度较少需要实例化的类的数量。如果能把那些参数移动到类实例的外面,在方法调用的时候将他们传递进来,就可以通过共享大幅度第减少单个实例 的数目。

那么如果在JavaScript中应用享元模式呢?有两种方式,第一种是应用在数据层上,主要是应用在内存里大量相似的对象上;第二种是应用在DOM层上,享元可以用在中央事件管理器上用来避免给父容器里的每个子元素都附加事件句柄。

享元与数据层

Flyweight中有两个重要概念--内部状态intrinsic和外部状态extrinsic之分,内部状态就是在对象里通过内部方法管理,而外部信息可以在通过外部删除或者保存。

说白点,就是先捏一个的原始模型,然后随着不同场合和环境,再产生各具特征的具体模型,很显然,在这里需要产生不同的新对象,所以Flyweight模式中常出现Factory模式,Flyweight的内部状态是用来共享的,Flyweight factory负责维护一个Flyweight pool(模式池)来存放内部状态的对象。

使用享元模式

让我们来演示一下如果通过一个类库让系统来管理所有的书籍,每个书籍的元数据暂定为如下内容:

ID

Title

Author

Genre

Page count

Publisher ID

ISBN

我们还需要定义每本书被借出去的时间和借书人,以及退书日期和是否可用状态:
checkoutDate

checkoutMember

dueReturnDate

availability

因为book对象设置成如下代码,注意该代码还未被优化:
var Book = function( id, title, author, genre, pageCount,publisherID, ISBN, checkoutDate, checkoutMember, dueReturnDate,availability ){

   this.id = id;

   this.title = title;

   this.author = author;

   this.genre = genre;

   this.pageCount = pageCount;

   this.publisherID = publisherID;

   this.ISBN = ISBN;

   this.checkoutDate = checkoutDate;

   this.checkoutMember = checkoutMember;

   this.dueReturnDate = dueReturnDate;

   this.availability = availability;

};

Book.prototype = {

   getTitle:function(){

       return this.title;

   },

   getAuthor: function(){

       return this.author;

   },

   getISBN: function(){

       return this.ISBN;

   },

/*其它get方法在这里就不显示了*/
// 更新借出状态

updateCheckoutStatus: function(bookID, newStatus, checkoutDate,checkoutMember, newReturnDate){

   this.id  = bookID;

   this.availability = newStatus;

   this.checkoutDate = checkoutDate;

   this.checkoutMember = checkoutMember;

   this.dueReturnDate = newReturnDate;

},

//续借

extendCheckoutPeriod: function(bookID, newReturnDate){

    this.id =  bookID;

    this.dueReturnDate = newReturnDate;

},

//是否到期

isPastDue: function(bookID){

   var currentDate = new Date();

   return currentDate.getTime() > Date.parse(this.dueReturnDate);

 }

};

程序刚开始可能没问题,但是随着时间的增加,图书可能大批量增加,并且每种图书都有不同的版本和数量,你将会发现系统变得越来越慢。几千个book对象在内存里可想而知,我们需要用享元模式来优化。

我们可以将数据分成内部和外部两种数据,和book对象相关的数据(title, author 等)可以归结为内部属性,而(checkoutMember, dueReturnDate等)可以归结为外部属性。这样,如下代码就可以在同一本书里共享同一个对象了,因为不管谁借的书,只要书是同一本书,基本信息是一样的:

/*享元模式优化代码*/

var Book = function(title, author, genre, pageCount, publisherID, ISBN){

   this.title = title;

   this.author = author;

   this.genre = genre;

   this.pageCount = pageCount;

   this.publisherID = publisherID;

   this.ISBN = ISBN;

};

定义基本工厂

让我们来定义一个基本工厂,用来检查之前是否创建该book的对象,如果有就返回,没有就重新创建并存储以便后面可以继续访问,这确保我们为每一种书只创建一个对象:

/* Book工厂 单例 */

var BookFactory = (function(){

   var existingBooks = {};

   return{

       createBook: function(title, author, genre,pageCount,publisherID,ISBN){

       /*查找之前是否创建*/

           var existingBook = existingBooks[ISBN];

           if(existingBook){

                   return existingBook;

               }else{

               /* 如果没有,就创建一个,然后保存*/

               var book = new Book(title, author, genre,pageCount,publisherID,ISBN);

               existingBooks[ISBN] =  book;

               return book;

           }

       }

   }

});

管理外部状态
外部状态,相对就简单了,除了我们封装好的book,其它都需要在这里管理:
/*BookRecordManager 借书管理类 单例*/

var BookRecordManager = (function(){

   var bookRecordDatabase = {};

   return{

       /*添加借书记录*/

       addBookRecord: function(id, title, author, genre,pageCount,publisherID,ISBN, checkoutDate, checkoutMember, dueReturnDate, availability){

           var book = bookFactory.createBook(title, author, genre,pageCount,publisherID,ISBN);

            bookRecordDatabase[id] ={

               checkoutMember: checkoutMember,

               checkoutDate: checkoutDate,

               dueReturnDate: dueReturnDate,

               availability: availability,

               book: book;
           };

       },

    updateCheckoutStatus: function(bookID, newStatus, checkoutDate, checkoutMember,     newReturnDate){

        var record = bookRecordDatabase[bookID];

        record.availability = newStatus;

        record.checkoutDate = checkoutDate;

        record.checkoutMember = checkoutMember;

        record.dueReturnDate = newReturnDate;

   },

   extendCheckoutPeriod: function(bookID, newReturnDate){

       bookRecordDatabase[bookID].dueReturnDate = newReturnDate;

   },

   isPastDue: function(bookID){

       var currentDate = new Date();

       return currentDate.getTime() > Date.parse(bookRecordDatabase[bookID].dueReturnDate);

   }

 };

});

通过这种方式,我们做到了将同一种图书的相同信息保存在一个bookmanager对象里,而且只保存一份;相比之前的代码,就可以发现节约了很多内存。

享元模式与DOM

关于DOM的事件冒泡,在这里就不多说了,相信大家都已经知道了,我们举两个例子。

例1:事件集中管理

举例来说,如果我们又很多相似类型的元素或者结构(比如菜单,或者ul里的多个li)都需要监控他的click事件的话,那就需要多每个元素进行事件绑定,如果元素有非常非常多,那性能就可想而知了,而结合冒泡的知识,任何一个子元素有事件触发的话,那触发以后事件将冒泡到上一级元素,所以利用这个特性,我们可以使用享元模式,我们可以对这些相似元素的父级元素进行事件监控,然后再判断里面哪个子元素有事件触发了,再进行进一步的操作。

在这里我们结合一下jQuery的bind/unbind方法来举例。

HTML:

<div id="container">

   <div class="toggle" href="#">更多信息 (地址)

       <span class="info">

          这里是更多信息

       </span></div>

   <div class="toggle" href="#">更多信息 (地图)

       <span class="info">

          <iframe src="http://www.map-generator.net/extmap.php?name=London&address=london%2C%20england&width=500...gt;"</iframe>

       </span>

   </div>

</div>

JavaScript:
stateManager = {

   fly: function(){

       var self =  this;

       $('#container').unbind().bind("click", function(e){

           var target = $(e.originalTarget || e.srcElement);

           // 判断是哪一个子元素

           if(target.is("div.toggle")){

               self.handleClick(target);

           }

       });

   },
   handleClick: function(elem){

       elem.find('span').toggle('slow');

   }

});

例2:应用享元模式提升性能

另外一个例子,依然和jQuery有关,一般我们在事件的回调函数里使用元素对象是会后,经常会用到$(this)这种形式,其实它重复创建了新对象,因为本身回调函数里的this已经是DOM元素自身了,我们必要必要使用如下这样的代码:

$('div').bind('click', function(){

 console.log('You clicked: ' + $(this).attr('id'));

});

// 上面的代码,要避免使用,避免再次对DOM元素进行生成jQuery对象,因为这里可以直接使用DOM元素自身了。

$('div').bind('click', function(){

 console.log('You clicked: ' + this.id);

});

其实,如果非要用$(this)这样的形式,我们也可以实现自己版本的单实例模式,比如我们来实现一个jQuery.signle(this)这样的函数以便返回DOM元素自身:
jQuery.single = (function(o){
   var collection = jQuery([1]);

   return function(element) {
       // 将元素放到集合里

       collection[0] = element;
        // 返回集合

       return collection;
   };

 });

 

使用方法:
$('div').bind('click', function(){

   var html = jQuery.single(this).next().html();

   console.log(html);

 });

这样,就是原样返回DOM元素自身了,而且不进行jQuery对象的创建。

总结

Flyweight模式是一个提高程序效率和性能的模式,会大大加快程序的运行速度.应用场合很多:比如你要从一个数据库中读取一系列字符串,这些字符串中有许多是重复的,那么我们可以将这些字符串储存在Flyweight池(pool)中。

如果一个应用程序使用了大量的对象,而这些大量的对象造成了很大的存储开心时就应该考虑使用享元模式;还有就是对象的大多数状态可以外部状态,如果删除对象的外部状态,那么就可以用相对较少的共享对象取代很多组对象,此时可以考虑使用享元模式。

Javascript 相关文章推荐
纯js实现背景图片切换效果代码
Nov 14 Javascript
javascript复制对象使用说明
Jun 28 Javascript
JavaScript 匿名函数(anonymous function)与闭包(closure)
Oct 04 Javascript
jQuery图片的展开和收缩实现代码
Apr 16 Javascript
js替代copy(示例代码)
Nov 27 Javascript
js 操作select与option(示例讲解)
Dec 20 Javascript
JavaScript的Backbone.js框架的一些使用建议整理
Feb 14 Javascript
移动端H5开发 Turn.js实现很棒的翻书效果
Jun 20 Javascript
如何在JS中实现相互转换XML和JSON
Jul 19 Javascript
使用Ajax与服务器(JSON)通信实例
Nov 04 Javascript
angular.fromJson与toJson方法用法示例
May 17 Javascript
基于Angular 8和Bootstrap 4实现动态主题切换的示例代码
Feb 11 Javascript
jQuery插件开发的五种形态小结
Mar 04 #Javascript
深入理解JavaScript系列(36):设计模式之中介者模式详解
Mar 04 #Javascript
百度UEditor编辑器如何关闭抓取远程图片功能
Mar 03 #Javascript
jQuery实现复选框成对选择及对应取消的方法
Mar 03 #Javascript
js实现文本框中输入文字页面中div层同步获取文本框内容的方法
Mar 03 #Javascript
JS实现文字放大效果的方法
Mar 03 #Javascript
jQuery实现的感应鼠标悬停图片色彩渐显效果
Mar 03 #Javascript
You might like
用PHP连接MySQL代码的参数说明
2008/06/07 PHP
js 动态添加标签(新增一行,其实很简单,就是几个函数的应用)
2009/03/26 Javascript
基于jquery自定义图片热区效果
2012/07/21 Javascript
原生javascript图片自动或手动切换示例附演示源码
2013/09/04 Javascript
javascript实现获取cookie过期时间的变通方法
2014/08/14 Javascript
jQuery插件expander实现图片翻转特效
2015/05/21 Javascript
Javascript基于AJAX回调函数传递参数实例分析
2015/12/15 Javascript
JavaScript高仿支付宝倒计时页面及代码实现
2016/10/21 Javascript
JavaScript 限制文本框不可输入英文单双引号的方法
2016/12/20 Javascript
Angular.js基础学习之初始化
2017/03/10 Javascript
微信小程序 图片上传实例详解
2017/05/05 Javascript
妙用Angularjs实现表格按指定列排序
2017/06/23 Javascript
js隐式转换的知识实例讲解
2018/09/28 Javascript
jQuery动态操作表单示例【基于table表格】
2018/12/06 jQuery
vue图片上传组件使用详解
2019/12/23 Javascript
[37:47]IG vs Winstrike 2018国际邀请赛小组赛BO2 第二场 8.19
2018/08/21 DOTA
python实现上传样本到virustotal并查询扫描信息的方法
2014/10/05 Python
Python开发中爬虫使用代理proxy抓取网页的方法示例
2017/09/26 Python
Python3数据库操作包pymysql的操作方法
2018/07/16 Python
Python使用线程来接收串口数据的示例
2019/07/02 Python
python批量修改ssh密码的实现
2019/08/08 Python
Python使用paramiko连接远程服务器执行Shell命令的实现
2021/03/04 Python
详解CSS3的box-shadow属性制作边框阴影效果的方法
2016/05/10 HTML / CSS
英国曼彻斯特宠物用品品牌:Bunty Pet Products
2019/07/27 全球购物
解释下面关于J2EE的名词
2013/11/15 面试题
行政文员岗位职责
2013/11/08 职场文书
三年级评语大全
2014/04/23 职场文书
减负增效提质方案
2014/05/23 职场文书
个人求职自荐信范文
2014/06/20 职场文书
领导班子专题民主生活会情况想汇报
2014/09/30 职场文书
技术股份合作协议书
2014/10/05 职场文书
学校总务处领导干部个人对照检查材料思想汇报
2014/10/06 职场文书
六一儿童节园长致辞
2015/07/31 职场文书
python爬虫之利用selenium模块自动登录CSDN
2021/04/22 Python
解读MySQL的客户端和服务端协议
2021/05/10 MySQL
前端canvas中物体边框和控制点的实现示例
2022/08/05 Javascript