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 file_exists问题杂谈
May 07 PHP
php中调用其他系统http接口的方法说明
Feb 28 PHP
ThinkPHP之A方法实例讲解
Jun 20 PHP
PHP导出Excel实例讲解
Jan 24 PHP
PHP获取昨天、今天及明天日期的方法
Feb 03 PHP
PHP编程计算文件或数组中单词出现频率的方法
May 22 PHP
Laravel如何创建服务器提供者实例代码
Apr 15 PHP
Laravel Validator自定义错误返回提示消息并在前端展示
May 09 PHP
PHP实现微信公众号验证Token的示例代码
Dec 16 PHP
thinkphp框架实现路由重定义简化url访问地址的方法分析
Apr 04 PHP
Yii使用EasyWechat实现小程序获取用户的openID的方法
Apr 29 PHP
如何在PHP环境中使用ProtoBuf数据格式
Jun 19 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
PHP5中MVC结构学习
2006/10/09 PHP
php empty() 检查一个变量是否为空
2011/11/10 PHP
thinkphp5.1 文件引入路径问题及注意事项
2018/06/13 PHP
mac pecl 安装php7.1扩展教程
2019/10/17 PHP
BOOM vs RR BO5 第二场 2.14
2021/03/10 DOTA
类似GMAIL的Ajax信息反馈显示
2010/02/16 Javascript
浅谈利用JavaScript进行的DDoS攻击原理与防御
2015/06/04 Javascript
JavaScript仿微博输入框效果(案例分析)
2016/12/06 Javascript
微信小程序 详解下拉加载与上拉刷新实现方法
2017/01/13 Javascript
Angularjs分页查询的实现
2017/02/24 Javascript
关于jquery form表单序列化的注意事项详解
2017/08/01 jQuery
nodeJS微信分享
2017/12/20 NodeJs
vue实现添加与删除图书功能
2018/10/07 Javascript
Vue源码解析之Template转化为AST的实现方法
2018/12/14 Javascript
VeeValidate 的使用场景以及配置详解
2019/01/11 Javascript
JavaScript刷新页面的几种方法总结
2019/03/28 Javascript
js实现表单项的全选、反选及删除操作示例
2020/06/05 Javascript
小程序中的箭头函数的具体使用
2020/06/19 Javascript
[04:50]2019DOTA2高校联赛秋季赛四强集锦
2019/12/27 DOTA
python中二维阵列的变换实例
2014/10/09 Python
python标准算法实现数组全排列的方法
2015/03/17 Python
python机器学习之贝叶斯分类
2018/03/26 Python
django.db.utils.ProgrammingError: (1146, u“Table‘’ doesn’t exist”)问题的解决
2018/07/13 Python
Django如何自定义分页
2018/09/25 Python
python实现飞机大战小游戏
2019/11/08 Python
python实现两个一维列表合并成一个二维列表
2019/12/02 Python
python suds访问webservice服务实现
2020/06/26 Python
使用python库xlsxwriter库来输出各种xlsx文件的示例
2020/09/01 Python
Python collections.deque双边队列原理详解
2020/10/05 Python
CSS3教程(5):网页背景图片
2009/04/02 HTML / CSS
女士时装鞋:Chinese Laundry
2018/08/29 全球购物
MYSQL基础面试题
2012/05/13 面试题
自我评价如何写好?
2014/01/05 职场文书
毕业自我鉴定总结
2014/03/24 职场文书
党员应该树立反腐倡廉的坚定意识思想汇报
2014/09/12 职场文书
Redis主从复制操作和配置详情
2022/09/23 Redis