php设计模式 FlyWeight (享元模式)


Posted in PHP onJune 26, 2011

享元模式英文称为“Flyweight Pattern”,我非常感谢将Flyweight Pattern翻译成享元模式的那位强人,因为这个词将这个模式使用的方式明白得表示了出来;如果翻译成为羽量级模式或者蝇量级模式等等,虽然可以含蓄的表现出使用此模式达到的目的,但是还是没有抓住此模式的关键。

享元模式的定义为:采用一个共享来避免大量拥有相同内容对象的开销。这种开销中最常见、直观的就是内存的损耗。享元模式以共享的方式高效的支持大量的细粒度对象。

在名字和定义中都体现出了共享这一个核心概念,那么怎么来实现共享呢?要知道每个事物都是不同的,但是又有一定的共性,如果只有完全相同的事物才能共享,那么享元模式可以说就是不可行的;因此我们应该尽量将事物的共性共享,而又保留它的个性。为了做到这点,享元模式中区分了内蕴状态和外蕴状态。内蕴状态就是共性,外蕴状态就是个性了。

注:共享的对象必须是不可变的,不然一变则全变(如果有这种需求除外)。

内蕴状态存储在享元内部,不会随环境的改变而有所不同,是可以共享的;外蕴状态是不可以共享的,它随环境的改变而改变的,因此外蕴状态是由客户端来保持(因为环境的变化是由客户端引起的)。在每个具体的环境下,客户端将外蕴状态传递给享元,从而创建不同的对象出来。

先看看下面程序,大概了解下享元模式。

<?php 
/** 
* 享元模式 
* 
* 运用享元技术有效的支持大量细粒度的对象 
*/ 
class CD 
{ 
private $_title = null; 
private $_artist = null; 
public function setTitle($title) 
{ 
$this->_title = $title; 
} 
public function getTitle() 
{ 
return $this->_title; 
} 
public function setArtist($artist) 
{ 
$this->_artist = $artist; 
} 
public function getArtist($artist) 
{ 
return $this->_artist; 
} 
} 
class Artist 
{ 
private $_name; 
public function __construct($name) 
{ 
echo "construct ".$name."<br/>"; 
$this->_name = $name; 
} 
public function getName() 
{ 
return $this->_name; 
} 
} 
class ArtistFactory 
{ 
private $_artists = array(); 
public function getArtist($name) 
{ 
if(isset($this->_artists[$name])) 
{ 
return $this->_artists[$name]; 
} else { 
$objArtist = new Artist($name); 
$this->_artists[$name] = $objArtist; 
return $objArtist; 
} 
} 
} 
$objArtistFactory = new ArtistFactory(); 
$objCD1 = new CD(); 
$objCD1->setTitle("title1"); 
$objCD1->setArtist($objArtistFactory->getArtist('artist1')); 
$objCD2 = new CD(); 
$objCD2->setTitle("title2"); 
$objCD2->setArtist($objArtistFactory->getArtist('artist2')); 
$objCD3 = new CD(); 
$objCD3->setTitle("title3"); 
$objCD3->setArtist($objArtistFactory->getArtist('artist1'));

享元模式的精要有三点:

  1. 被系统大量使用的细粒度对象,粒度要有多细,量要有多大,看看jdk中使用的享元模式就知道了,jdk中,Integer,Character,String等都使用了享元模式,他们都是最基础的数据类型,不可谓不细,他们频繁的参与运算,不可谓不大量。
  2. 划分对象的内蕴属性/状态和外蕴属性/状态;所谓内蕴状态,就是存在对象的内部,不会随着环境变化的状态, 有一个网友说的很好,就是无区别的状态, 即拿掉外蕴属性之后同一类对象没有区别对象的内蕴状态就是对象的元神,只要元神元神无区别,那么对象也就无区别,同时也只有这些无区别的元神可以被共享,我想这也是Flyweight被翻译成享元的原因。外蕴状态就是由客户端指定,会随着环境变化的状态; 对于Integer来说, 他的内蕴属性其实就是他的value(当然它也没有外蕴属性);
  3. 用一个工厂控制享元的创造;因为享元对象不能被客户端随意创造, 否则就没有意义了。工厂通常提供缓存机制保存已经创造的享元。

面向对象虽然很好地解决了抽象性的问题,但是对于一个实际运行的软件系统,我们还需要考虑面向对象的代价问题,享元模式解决的就是面向对象的代价问题。享元模式采用对象共享的做法来降低系统中对象的个数,从而降低细粒度对象给系统带来的内存压力。

享元模式在一般的项目开发中并不常用,而是常常应用于系统底层的开发,以便解决系统的性能问题。Java和.Net中的String类型就是使用了享元模式。如果在Java或者.NET中已经创建了一个字符串对象s1,那么下次再创建相同的字符串s2的时候,系统只是把s2的引用指向s1所引用的具体对象,这就实现了相同字符串在内存中的共享。如果每次执行s1=“abc”操作的时候,都创建一个新的字符串对象的话,那么内存的开销会很大。

PHP 相关文章推荐
使用PHP批量生成随机用户名
Jul 10 PHP
用mysql内存表来代替php session的类
Feb 01 PHP
在PHP中养成7个面向对象的好习惯
Jan 28 PHP
php下获取客户端ip地址的函数
Mar 15 PHP
PHP时间格式控制符对照表分享
Jul 23 PHP
PHP实现数字补零功能的2个函数介绍
May 12 PHP
php获取从html表单传递数组的方法
Mar 20 PHP
php实现的后台表格分页功能示例
Oct 23 PHP
PHP封装类似thinkphp连贯操作数据库Db类与简单应用示例
May 08 PHP
PHP+redis实现微博的推模型案例分析
Jul 10 PHP
php curl发送请求实例方法
Aug 01 PHP
laravel与thinkphp之间的区别与优缺点
Mar 02 PHP
php设计模式 Mediator (中介者模式)
Jun 26 #PHP
php设计模式 Prototype (原型模式)代码
Jun 26 #PHP
PHP如何解决网站大流量与高并发的问题
Jun 25 #PHP
session在PHP大型web应用中的使用
Jun 25 #PHP
php URL跳转代码 减少外链
Jun 25 #PHP
php session安全问题分析
Jun 24 #PHP
使用PHP实现二分查找算法代码分享
Jun 24 #PHP
You might like
用Socket发送电子邮件(利用需要验证的SMTP服务器)
2006/10/09 PHP
php _autoload自动加载类与机制分析
2012/02/10 PHP
WordPress中限制非管理员用户在文章后只能评论一次
2015/12/31 PHP
php从数据库中获取数据用ajax传送到前台的方法
2018/08/20 PHP
PHP全局使用Laravel辅助函数dd
2019/12/26 PHP
JavaScript 设计模式学习 Singleton
2009/07/27 Javascript
jquery tablesorter.js 支持中文表格排序改进
2009/12/09 Javascript
JQuery 操作select标签实现代码
2010/05/14 Javascript
jquery ajax post提交数据乱码
2013/11/05 Javascript
js实现简单锁屏功能实例
2015/05/27 Javascript
javascript实现查找数组中最大值方法汇总
2016/02/13 Javascript
Vue表单验证插件Vue Validator使用方法详解
2017/04/07 Javascript
AngularJS实现动态添加Option的方法
2017/05/17 Javascript
详解Vuex中mapState的具体用法
2017/09/28 Javascript
JS随机排序数组实现方法分析
2017/10/11 Javascript
原生JS+HTML5实现跟随鼠标一起流动的粒子动画效果
2018/05/03 Javascript
Angular angular-file-upload文件上传的示例代码
2018/08/23 Javascript
微信小程序 如何获取网络状态
2019/07/26 Javascript
jquery实现简单自动轮播图效果
2020/07/29 jQuery
vue实现防抖的实例代码
2021/01/11 Vue.js
Python正确重载运算符的方法示例详解
2017/08/27 Python
Python实现螺旋矩阵的填充算法示例
2017/12/28 Python
python自动截取需要区域,进行图像识别的方法
2018/05/17 Python
对Python实现简单的API接口实例讲解
2018/12/10 Python
python config文件的读写操作示例
2019/09/27 Python
python numpy 矩阵堆叠实例
2020/01/17 Python
pandas 像SQL一样使用WHERE IN查询条件说明
2020/06/05 Python
使paramiko库执行命令时在给定的时间强制退出功能的实现
2021/03/03 Python
HTML5+CSS3应用详解
2014/02/24 HTML / CSS
详解淘宝H5 sign加密算法
2020/08/25 HTML / CSS
酒店工作职员求职简历的自我评价
2013/10/23 职场文书
会计专业毕业生自荐信范文
2013/12/20 职场文书
校园奶茶店创业计划书
2014/01/23 职场文书
竞选部长演讲稿
2014/04/26 职场文书
设计顾问服务计划书
2014/05/04 职场文书
2015新学期开学寄语
2015/02/26 职场文书