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 相关文章推荐
php中iconv函数使用方法
May 24 PHP
linux iconv方法的使用
Oct 01 PHP
win7+apache+php+mysql环境配置操作详解
Jun 10 PHP
Thinkphp实现MySQL读写分离操作示例
Jun 25 PHP
laravel安装zend opcache加速器教程
Mar 02 PHP
PHP连接操作access数据库实例
Mar 30 PHP
UTF-8正则表达式如何匹配汉字
Aug 03 PHP
PHP引用返回用法示例
May 28 PHP
PHP Ajax实现无刷新附件上传
Aug 17 PHP
微信公众号开发之通过接口删除菜单
Feb 20 PHP
PHP实现图片的等比缩放和Logo水印功能示例
May 04 PHP
CakePHP框架Model关联对象用法分析
Aug 04 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
PHP学习之PHP变量
2006/10/09 PHP
一个简单且很好用的php分页类
2013/10/26 PHP
thinkphp实现图片上传功能分享
2014/03/04 PHP
windows平台中配置nginx+php环境
2015/12/06 PHP
Zend Framework动作助手Url用法详解
2016/03/05 PHP
PHP PDOStatement::errorCode讲解
2019/01/31 PHP
繁简字转换功能
2006/07/19 Javascript
JQuery中getJSON的使用方法
2010/12/13 Javascript
公共js在页面底部加载的注意事项介绍
2013/07/18 Javascript
jsPDF导出pdf示例
2014/05/02 Javascript
html的DOM中Event对象onabort事件用法实例
2015/01/21 Javascript
Angularjs制作简单的路由功能demo
2015/04/14 Javascript
JavaScript中的getDay()方法使用详解
2015/06/09 Javascript
jquery地址栏链接与a标签链接匹配之特效代码总结
2015/08/24 Javascript
如何防止JavaScript自动插入分号
2015/11/05 Javascript
js表单中选择框值的获取及表单的序列化
2015/12/17 Javascript
原生javascript实现的一个简单动画效果
2016/03/30 Javascript
javascript中Number的方法小结
2016/11/21 Javascript
JavaScript数据结构之二叉树的删除算法示例
2017/04/13 Javascript
AngularJS 支付倒计时功能实现思路
2017/06/05 Javascript
vue 父组件调用子组件方法及事件
2018/03/29 Javascript
vue.extend实现alert模态框弹窗组件
2018/04/28 Javascript
微信小程序实现刷脸登录
2018/05/25 Javascript
javascript刷新父页面方法汇总详解
2019/10/10 Javascript
ES6新增的数组知识实例小结
2020/05/23 Javascript
Javascript实现关闭广告效果
2021/01/29 Javascript
python处理图片之PIL模块简单使用方法
2015/05/11 Python
Python解析json之ValueError: Expecting property name enclosed in double quotes: line 1 column 2(char 1)
2017/07/06 Python
在keras中对单一输入图像进行预测并返回预测结果操作
2020/07/09 Python
Html5导航栏吸顶方案原理与对比实现
2020/06/10 HTML / CSS
澳洲CFL商城:CHEMIST FOR LESS(中文)
2021/02/28 全球购物
在职人员函授期间自我评价分享
2013/11/08 职场文书
平民服装店创业计划书
2014/01/17 职场文书
三查三看党性分析材料
2014/02/18 职场文书
vue+elementui 实现新增和修改共用一个弹框的完整代码
2021/06/08 Vue.js
MySQL性能指标TPS+QPS+IOPS压测
2022/08/05 MySQL