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程序--记数器
Oct 09 PHP
PHP教程 预定义变量
Oct 23 PHP
php的ajax框架xajax入门与试用介绍
Dec 19 PHP
php文件操作实例代码
May 10 PHP
解析php session_set_save_handler 函数的用法(mysql)
Jun 29 PHP
jQuery+PHP实现的掷色子抽奖游戏实例
Jan 04 PHP
php中preg_match的isU代表什么意思
Oct 01 PHP
Smarty环境配置与使用入门教程
May 11 PHP
PHP实现的自定义数组排序函数与排序类示例
Nov 18 PHP
PHP进程通信基础之信号量与共享内存通信
Feb 19 PHP
PHP设计模式之模板方法模式定义与用法详解
Apr 02 PHP
PHP实现的文件浏览器功能简单示例
Sep 12 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
Yii框架批量插入数据扩展类的简单实现方法
2017/05/23 PHP
基于Laravel 5.2 regex验证的正确写法
2019/09/29 PHP
offsetParent 算法分析
2010/04/05 Javascript
javascript 函数使用说明
2010/04/07 Javascript
Javascript级联下拉菜单以及AJAX数据验证核心代码
2013/05/10 Javascript
判定是否原生方法的JS代码
2013/11/12 Javascript
javascript写的一个模拟阅读小说的程序
2014/04/04 Javascript
js实现带圆角的两级导航菜单效果代码
2015/08/24 Javascript
JS实现双击屏幕滚动效果代码
2015/10/28 Javascript
mvc中form表单提交的三种方式(推荐)
2016/08/10 Javascript
移动端js触摸事件详解
2016/09/18 Javascript
百度多文件异步上传控件webuploader基本用法解析
2016/11/07 Javascript
ECMAScript6 新特性范例大全
2017/03/24 Javascript
jquery 禁止鼠标右键并监听右键事件
2017/04/27 jQuery
JavaScript输入框字数实时统计更新
2017/06/17 Javascript
JS html事件冒泡和事件捕获操作示例
2019/05/01 Javascript
Vue项目中配置pug解析支持
2019/05/10 Javascript
JavaScript深入V8引擎以及编写优化代码的5个技巧
2019/06/24 Javascript
[34:41]夜魇凡尔赛茶话会 第二期02:你画我猜
2021/03/11 DOTA
Python 专题五 列表基础知识(二维list排序、获取下标和处理txt文本实例)
2017/03/20 Python
详解Python函数可变参数定义及其参数传递方式
2017/08/02 Python
python与sqlite3实现解密chrome cookie实例代码
2018/01/20 Python
Python语言进阶知识点总结
2019/05/28 Python
Python爬取破解无线网络wifi密码过程解析
2019/09/17 Python
Python3 中作为一等对象的函数解析
2019/12/11 Python
Python字典深浅拷贝与循环方式方法详解
2020/02/09 Python
Python如何把Spark数据写入ElasticSearch
2020/04/18 Python
Bergfreunde丹麦:登山装备网上零售商
2017/02/26 全球购物
敬老院院长事迹材料
2014/05/21 职场文书
开服装店计划书
2014/08/15 职场文书
邹越感恩父母演讲稿
2014/08/28 职场文书
吃空饷专项整治方案
2014/10/27 职场文书
后勤个人工作总结
2015/02/28 职场文书
告知书格式
2015/07/01 职场文书
2015年学校教研室主任工作总结
2015/07/20 职场文书
标准演讲稿格式结尾应该怎么书写?
2019/07/17 职场文书