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下检测字符串是否是utf8编码的代码
Jun 28 PHP
PHP SPL使用方法和他的威力
Nov 12 PHP
PHP获取当前所在目录位置的方法
Nov 26 PHP
php限制上传文件类型并保存上传文件的方法
Mar 13 PHP
php实现约瑟夫问题的方法小结
Mar 23 PHP
ThinkPHP中使用Ueditor富文本编辑器
Sep 02 PHP
Yii的Srbac插件用法详解
Jul 14 PHP
[原创]PHP正则匹配中英文、数字及下划线的方法【用户名验证】
Aug 01 PHP
详解thinkphp中的volist标签
Jan 15 PHP
PHP快速排序算法实现的原理及代码详解
Apr 03 PHP
Laravel 5.4前后台分离,通过不同的二级域名访问方法
Oct 13 PHP
PHP实现页面静态化深入讲解
Mar 04 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
如何使用PHP中的字符串函数
2006/11/24 PHP
mysql5写入和读出乱码解决
2006/11/25 PHP
php执行sql语句的写法
2009/03/10 PHP
Symfony2获取web目录绝对路径、相对路径、网址的方法
2016/11/14 PHP
使用Git实现Laravel项目的自动化部署
2019/11/24 PHP
用jQuery技术实现Tab页界面之二
2009/09/21 Javascript
关于Mozilla浏览器不支持innerText的解决办法
2011/01/01 Javascript
js限制checkbox选中个数以限制六个为例
2014/07/15 Javascript
jQuery实现的多级下拉菜单效果代码
2015/08/24 Javascript
原生js实现无缝轮播图效果
2017/01/11 Javascript
小程序获取周围IBeacon设备的方法
2018/10/31 Javascript
基于JS实现web端录音与播放功能
2019/04/17 Javascript
Ant Design Pro 下实现文件下载的实现代码
2019/12/03 Javascript
小程序表单认证布局及验证详解
2020/06/19 Javascript
微信小程序用户登录和登录态维护的实现
2020/12/10 Javascript
用Python程序抓取网页的HTML信息的一个小实例
2015/05/02 Python
简析Python的闭包和装饰器
2016/02/26 Python
Django的信号机制详解
2017/05/05 Python
Python2.7基于淘宝接口获取IP地址所在地理位置的方法【测试可用】
2017/06/07 Python
利用Python如何生成便签图片详解
2018/07/09 Python
Python turtle库绘制菱形的3种方式小结
2019/11/23 Python
tensorflow实现二维平面模拟三维数据教程
2020/02/11 Python
Python GUI编程学习笔记之tkinter事件绑定操作详解
2020/03/30 Python
python如何判断IP地址合法性
2020/04/05 Python
python中常见错误及解决方法
2020/06/21 Python
Theory美国官网:后现代都市风时装品牌
2018/05/09 全球购物
Rosetta Stone官方网站:语言学习
2019/01/05 全球购物
Turnbull & Asser官网:英国皇室御用的顶级定制衬衫
2019/01/31 全球购物
简述Linux文件系统通过i节点把文件的逻辑结构和物理结构转换的工作过程
2012/04/17 面试题
项目经理岗位职责
2013/11/11 职场文书
共产党员公开承诺书范文
2014/03/28 职场文书
业务员岗位职责
2015/02/03 职场文书
普宁寺导游词
2015/02/04 职场文书
2015年建筑工程工作总结
2015/05/13 职场文书
收入证明怎么写
2015/06/12 职场文书
Pytest之测试命名规则的使用
2021/04/16 Python