PHP设计模式之装饰者模式


Posted in PHP onFebruary 29, 2012

介绍
装饰者模式动态地将责任附加到对象上。若要扩展功能,装饰者提供了比继承更有弹性的替代方案。

思维导图

PHP设计模式之装饰者模式

 

有这样一个项目,做一个餐厅订餐系统。起初的代码结构是这样的。前面有很多Beverage的继承类,现在遇到的问题是牛奶的价钱上涨了,那么所有相关的类,我们都要进行调整,比如Milk,SugarAndMilk类,这种类还有很多,我们需要逐个去修改类中的方法——开发人员每次都做这种事情,要疯了!所以我们要改变现有的结构。以下的图都是简图,实际的图,可没有这么简单。

 

PHP设计模式之装饰者模式

 

 设计问题:

1》类数量爆炸,有很多类,难以维护;
2》整个设计呆板;
3》基类加入的新功能无法使用于子类;
复用类方法的方式很多,比如继承,组合,委托。为什么老是习惯用继承呢?我看Zend Framework也有这种习惯!每次找对应方法,一直往上翻。——题外话!!!!
后来经过小组研究决定,我们决定把基础类抽出来,比如,我们把咖啡做成一个单独的类,其他的咖啡,比如牛奶咖啡,甜味咖啡,我们只对材料单独包装成一个类。
经过改良的设计:

PHP设计模式之装饰者模式

详解
1》对于饮品,我们直接继承Beverage类,直接把报价写进饮品类里面;
2》而对于一些需要添加调味品的特殊饮品,我们做累加操作。比如,我想要杯奶咖啡,则 总价=咖啡价+奶价
3》这样不同的饮料就很容易知道它的价格。
代码

<?php 
abstract class Beverage{ 
public $_name; 
abstract public function Cost(); 
} 
// 被装饰者类 
class Coffee extends Beverage{ 
public function __construct(){ 
$this->_name = 'Coffee'; 
} 
public function Cost(){ 
return 1.00; 
} 
} 
// 以下三个类是装饰者相关类 
class CondimentDecorator extends Beverage{ 
public function __construct(){ 
$this->_name = 'Condiment'; 
} 
public function Cost(){ 
return 0.1; 
} 
} 
class Milk extends CondimentDecorator{ 
public $_beverage; 
public function __construct($beverage){ 
$this->_name = 'Milk'; 
if($beverage instanceof Beverage){ 
$this->_beverage = $beverage; 
}else 
exit('Failure'); 
} 
public function Cost(){ 
return $this->_beverage->Cost() + 0.2; 
} 
} 
class Sugar extends CondimentDecorator{ 
public $_beverage; 
public function __construct($beverage){ 
$this->_name = 'Sugar'; 
if($beverage instanceof Beverage){ 
$this->_beverage = $beverage; 
}else{ 
exit('Failure'); 
} 
} 
public function Cost(){ 
return $this->_beverage->Cost() + 0.2; 
} 
} 
// Test Case 
//1.拿杯咖啡 
$coffee = new Coffee(); 
//2.加点牛奶 
$coffee = new Milk($coffee); 
//3.加点糖 
$coffee = new Sugar($coffee); 
printf("Coffee Total:%0.2f元\n",$coffee->Cost());

总结
1.装饰者(Milk)和被装饰者(Coffee)必须是一样的类型。目的是装饰者必须取代被装饰者。
2.添加行为:当装饰者和组件组合时,就是在加入新的行为。
题外话:
1.利用继承设计子类行为,是在编译时静态决定的,而且所有的子类都会继承到相同的行为。打个比方,老子想学点功夫,看你小子会太极拳,老子只需要继承你一下 ,老子也就会太极拳了——呵呵,这时老子就变成你儿子了,看来继承是要付出代价的。
2.组合,我们可以扩展对象的行为,在运行时动态地进行扩展。利用组合我们可以随时把我们当时设计超类时没有想到的方法加入到对象中,而不用改变现有的代码。打个比方,老子现在没有内力,吸功大法,把和尚,尼姑,道士的内力(行为对象)都吸过来,那在搏斗(运行时)中,老子可以随时都能使用不同的内力,但也不能胡乱吸内力,否则你就要走火入魔了!
3.类应该对扩展开放,对修改关闭。如果我们每个部分都用装饰者模式进行设计,那么对于整个框架来说有点浪费,而且你也加大了代码的难度。那什么时候使用这种模式呢?我们一般用于经常改变的地方。那我们又怎么知道哪些是经常改变的地方呢?这个就需要我们的经验和你对所处行业的了解。建议大家平时多看点例子。
4.装饰模式为设计注入弹性,但同时会在设计中加入大量的小类,这偶尔会导致别人不容易了解这种设计。
5.在使用装饰者模式的时候,对插入的的装饰者要特别小心。因为装饰者模式依赖某种特定的类型(Beverage)。
6.要想很好的使用装饰者模式,我们还要配合使用工厂模式和生成器模式,但今天只说装饰者模式。要想知道更多,请听下回分解。
参考文献:《head first 设计模式》
PHP 相关文章推荐
PHP5在Apache下的两种模式的安装
Sep 05 PHP
PHP的宝库目录--PEAR
Oct 09 PHP
PHP4实际应用经验篇(2)
Oct 09 PHP
基于php冒泡排序算法的深入理解
Jun 09 PHP
php函数间的参数传递(值传递/引用传递)
Sep 23 PHP
跨浏览器PHP下载文件名中的中文乱码问题解决方法
Mar 05 PHP
php字符串函数学习之strstr()
Mar 27 PHP
php实现搜索类封装示例
Mar 31 PHP
利用php_imagick实现复古效果的方法
Oct 18 PHP
php使用curl实现ftp文件下载功能
May 16 PHP
PHP实现深度优先搜索算法(DFS,Depth First Search)详解
Sep 16 PHP
PHP实现找出链表中环的入口节点
Jan 16 PHP
php preg_filter执行一个正则表达式搜索和替换
Feb 27 #PHP
mysql总结之explain
Feb 27 #PHP
php&amp;mysql 日期操作小记
Feb 27 #PHP
MySQL时间字段究竟使用INT还是DateTime的说明
Feb 27 #PHP
php explode函数实例代码
Feb 27 #PHP
PHP中获取文件扩展名的N种方法小结
Feb 27 #PHP
PHP中的正则表达式函数介绍
Feb 27 #PHP
You might like
thinkphp实现上一篇与下一篇的方法
2014/12/08 PHP
php计算2个日期的差值函数分享
2015/02/02 PHP
php实现阿拉伯数字和罗马数字相互转换的方法
2015/04/17 PHP
CodeIgniter生成静态页的方法
2016/05/17 PHP
PHP的AES加密算法完整实例
2016/07/20 PHP
PHP精确计算功能示例
2016/11/29 PHP
ThinkPHP下表单令牌错误与解决方法分析
2017/05/20 PHP
跟着JQuery API学Jquery 之二 属性
2010/04/09 Javascript
jquery validate使用攻略 第四步
2010/07/01 Javascript
jQuery(非HTML5)可编辑表格实现代码
2012/12/11 Javascript
清除div下面的所有标签的方法
2014/02/17 Javascript
jQuery 写的简单打字游戏可以提示正确和错误的次数
2014/07/01 Javascript
jQuery插件实现大图全屏图片相册
2015/03/14 Javascript
javascript时间排序算法实现活动秒杀倒计时效果
2021/01/28 Javascript
浅谈JavaScript的函数及作用域
2016/12/30 Javascript
详解Node.js项目APM监控之New Relic
2017/05/12 Javascript
移动端手指放大缩小插件与js源码
2017/05/22 Javascript
关于vue.extend和vue.component的区别浅析
2017/08/16 Javascript
vue 每次渲染完页面后div的滚动条保持在最底部的方法
2018/03/17 Javascript
angular 实现的输入框数字千分位及保留几位小数点功能示例
2018/06/19 Javascript
Vue表单demo v-model双向绑定问题
2018/06/29 Javascript
详解如何在Node.js的httpServer中接收前端发送的arraybuffer数据
2018/11/11 Javascript
Vue的路由及路由钩子函数的实现
2019/07/02 Javascript
vuex实现数据状态持久化
2019/11/11 Javascript
node.js中对Event Loop事件循环的理解与应用实例分析
2020/02/14 Javascript
Node.js Domain 模块实例详解
2020/03/18 Javascript
js实现鼠标拖曳效果
2020/12/30 Javascript
linux环境下的python安装过程图解(含setuptools)
2017/11/22 Python
Python列表生成式与生成器操作示例
2018/08/01 Python
keras 解决加载lstm+crf模型出错的问题
2020/06/10 Python
css3实现二维码扫描特效的示例
2020/10/29 HTML / CSS
印度尼西亚最完整和最大的在线药房网站:Farmaku.com
2019/11/23 全球购物
酒店销售经理岗位职责
2014/01/31 职场文书
教师中国梦演讲稿
2014/04/23 职场文书
餐饮服务食品安全承诺书
2015/04/29 职场文书
用Python简陋模拟n阶魔方
2021/04/17 Python