工厂模式告一段落,我们来研究其他一些模式。不知道各位大佬有没有尝试过女装?据说女装大佬程序员很多哟。其实,今天的装饰器模式就和化妆这件事很像。相信如果有程序媛MM在的话,马上就能和你讲清楚这个设计模式。

Gof类图及解释

装饰这两个字,我们暂且把他变成化妆。首先你得有一张脸,然后打底,然后上妆,可以早上来个淡妆上班,也可以下班的时候补成浓妆出去嗨。当然,码农们下班的时间点正好是能赶上夜场的下半场的。话说回来,不管怎么化妆,你的脸还是你的脸,有可能可以化成别人不认识的另一个人,但这的的确确还是你的脸。这就是装饰器,对对象(脸)进行各种装饰(化妆),让这个脸更好看(增加职责)。

GoF定义:动态地给一个对象添加一些额外的职责,就增加功能来说,Decorator模式相比生成子类更为灵活

GoF类图

代码实现

interface Component{public function operation();
}class ConcreteComponent implements Component{public function operation(){echo "I'm face!" . PHP_EOL;}
}

很简单的一个接口和一个实现,这里我们就把具体的实现类看作是一张脸吧!

abstract class Decorator implements Component{protected $component;public function __construct(Component $component){$this->component = $component;}
}

抽象的装饰者类,实现Component接口,但并不实现operation()方法,让子类去实现。在这里主要保存一个Componet的引用,一会就要对他进行装饰。对应到上方的具体类,我们就是要准备给脸化妆啦!

class ConcreteDecoratorA extends Decorator{public $addedState = 1; // 没什么实际意义的属性,只是区别于ConcreteDecoratorBpublic function operation(){echo $this->component->operation() . "Push " . $this->addedState . " cream!" . PHP_EOL;}
}
class ConcreteDecoratorB extends Decorator{public function operation(){$this->component->operation();$this->addedBehavior();}// 没什么实际意义的方法,只是区别于ConcreteDecoratorApublic function addedBehavior(){echo "Push 2 cream!" . PHP_EOL;}
}

两个具体装饰者。在这里我是涂了两次霜,毕竟是纯爷们,对化妆这事儿真的是不了解。好像第一步应该先是打粉底吧?不过这次就这样,我们这两个装饰器实现的就是给脸上涂两层霜。

  • 从代码中可以看出,我们是一直对具体的那个ConcreteComponent对象来进行包装
  • 再往下的话其实我们是对他的operation()这个方法包装了两次,每次都是在前一次的基础上加了一点点东西
  • 不要纠结于A和B装饰器上的added属性和方法,他们只是GoF类图中用以区别这两个装饰器不是同一个东西,每个装饰器都可以干很多别的事,Component对象也不一定只有operation()这一个方法,我们可以选择性的去装饰对象中的全部或者部分方法
  • 好像我们都继承了Component,直接子类一路重写不就行了,搞这费劲干嘛?亲,了解下组合的概念哟,我们的Decorator父类里面是一个真实对象的引用哦,解耦了自身哦,我们只给真实的对象去做包装,您可别直接实例化装饰器来直接用
  • 还是没懂?好处呢?老系统的类啊、方法啊你敢随便乱改?想给前任写的牛(S)逼(B)代码扩展新功能时不妨试试装饰器这货,说不定有奇效!

手机这玩意干不过某米、某O、某为,这没法玩呀,好吧,哥们去专心做手机壳吧!嗯,我先准备了一个透明壳(Component),貌似有点丑,没办法,谁叫哥们穷。给某米的加上各种纯色(DecoratorA1),然后背后印上各种颜色的植物(DecoratorB1)吧;某O的手机最近喜欢找流量明显做代言,那我给他的手机壳就用各种炫彩色(DecoratorA2)和明星的卡通头像(DecoratorB2);最后的某为,好像手机已经开始引领业界潮流了,折叠屏这玩意不是要砸我这卖手机壳的生意嘛!!好吧,哥不给你们做了,还是跟我的某米、某O混去吧!!

完整代码:装饰器模式

实例

继续来发短信,之前我们用工厂模式解决了多个短信运营商的问题。这回我们要解决的是短信内容模板的问题。对于推广类的短信来说,根据最新的广告法,我们是不能出现“全国第一”、“全世界第一”这类的词语的,当然,一些不太文明的用语我们也是不能使用的。

现在的情况是这样的,我们有一个很早之前的短信模板类,里面的内容是固定的,老系统依然还是使用这个模板,老系统是面对的内部员工,对语言内容的要求不高。而新系统则需要向全网发送,也就是内外部的用户都要发送。这时,我们可以用装饰器模式来对老系统的短信模板进行包装。其实说简单点,我们就是用装饰器来做文本替换的功能。好处呢?当然是可以不去改动原来的模板类中的方法就实现了对老模板内容的修改扩展等。

短信发送类图

完整源码:短信发送装饰器方法

<?php
// 短信模板接口
interface MessageTemplate
{public function message();
}// 假设有很多模板实现了上面的短信模板接口
// 下面这个是其中一个优惠券发送的模板实现
class CouponMessageTemplate implements MessageTemplate
{public function message(){return '优惠券信息:我们是全国第一的牛X产品哦,送您十张优惠券!';}
}// 我们来准备好装饰上面那个过时的短信模板
abstract class DecoratorMessageTemplate implements MessageTemplate
{public $template;public function __construct($template){$this->template = $template;}
}// 过滤新广告法中不允许出现的词汇
class AdFilterDecoratorMessage extends DecoratorMessageTemplate
{public function message(){return str_replace('全国第一', '全国第二', $this->template->message());}
}// 使用我们的大数据部门同事自动生成的新词库来过滤敏感词汇,这块过滤不是强制要过滤的内容,可选择使用
class SensitiveFilterDecoratorMessage extends DecoratorMessageTemplate
{public $bigDataFilterWords = ['牛X'];public $bigDataReplaceWords = ['好用'];public function message(){return str_replace($this->bigDataFilterWords, $this->bigDataReplaceWords, $this->template->message());}
}// 客户端,发送接口,需要使用模板来进行短信发送
class Message
{public $msgType = 'old';public function send(MessageTemplate $mt){// 发送出去咯if ($this->msgType == 'old') {echo '面向内网用户发送' . $mt->message() . PHP_EOL;} else if ($this->msgType == 'new') {echo '面向全网用户发送' . $mt->message() . PHP_EOL;}}
}$template = new CouponMessageTemplate();
$message = new Message();// 老系统,用不着过滤,只有内部用户才看得到
$message->send($template);// 新系统,面向全网发布的,需要过滤一下内容哦
$message->msgType = 'new';
$template = new AdFilterDecoratorMessage($template);
$template = new SensitiveFilterDecoratorMessage($template);// 过滤完了,发送吧
$message->send($template);

说明

  • 装饰器的最大好处:一是不改变原有代码的情况下对原有代码中的内容进行扩展,开放封闭原则;二是每个装饰器完成自己的功能,单一职责;三是用组合实现了继承的感觉;
  • 最适用于:给老系统进行扩展
  • 要小心:过多的装饰者会把你搞晕的
  • 不一定都是对同一个方法进行装饰,其实装饰者应该更多的用于对对象的装饰,对对象进行扩展,这里我们都是针对一个方法的输出进行装饰,但仅限此文,装饰器的应用其实更加广泛
  • 装饰器的特点是全部都继承自一个主接口或类,这样的好处就是返回的对象是相同的抽象数据,具有相同的行为属性,否则,就不是装饰之前的对象,而是一个新对象了
  • 有点不好理解没关系,我们这次的例子其实也很勉强,这个设计模式在《Head First设计模式》中有提到Java的I/O系列接口是使用的这种设计模式:FileInputStream、LineNumberInputStream、BufferInputStream等
  • Laravel框架中的中间件管道,这里其实是多种模式的综合应用,其中也应用到了装饰器模式:Laravel HTTP——Pipeline 中间件装饰者模式源码分析
  • 另外在Laravel中,日志处理这里也是对Monolog进行了装饰,有兴趣的同学可以去了解下

下期看点

又是大伽驾到,电源适配器了解吧?变压器总见过吧?你可能用过,也可能没用过,但你一定听说过这个非常非常出名的适配器模式

===============

关注公众号:【硬核项目经理】获取最新文章

添加微信/QQ好友:【DarkMatterZyCoder/149844827】免费得PHP、项目管理学习资料

PHP设计模式之装饰器模式相关推荐

  1. 设计模式学习----装饰器模式

    这两天本来是自在学习java collection Framework的Fail Fast底层机制,看到核心的部分时,突然意识到设计模式的问题,上大学到现在我还没有真正理解过设计模式的概念,于是用了大 ...

  2. 【设计模式】装饰器模式的使用

    问题来源 我们在进行软件系统设计的时候,有一些业务(如下图,一些通用的非功能性需求)是多个模块都需要的,是跨越模块的.把它们放到什么地方呢? 最简单的办法就是把这些通用模块的接口写好,让程序员在实现业 ...

  3. go设计模式之装饰器模式

    go设计模式之装饰器模式 再写这篇文章时,我已经看了很多其他人发表的类似文章,大概看了这么多吧. 亓斌的设计模式-装饰者模式(Go语言描述) jeanphorn的Golang设计模式之装饰模式 七八月 ...

  4. python中的装饰器、装饰器模式_python 设计模式之装饰器模式 Decorator Pattern

    #写在前面 已经有一个礼拜多没写博客了,因为沉醉在了<妙味>这部小说里,里面讲的是一个厨师苏秒的故事.现实中大部分人不会有她的天分.我喜欢她的性格:总是想着去解决问题,好像从来没有怨天尤人 ...

  5. python 设计模式之装饰器模式 Decorator Pattern

    #写在前面 已经有一个礼拜多没写博客了,因为沉醉在了<妙味>这部小说里,里面讲的是一个厨师苏秒的故事.现实中大部分人不会有她的天分.我喜欢她的性格:总是想着去解决问题,好像从来没有怨天尤人 ...

  6. 零基础学习设计模式之装饰器模式(配套视频)

    零基础学习设计模式之装饰器模式 定义 在不改变目标结构的情况下,动态的给对象增加功能 举例 如房子装修.相片加相框等,都是装饰器模式. 基本组件 抽象构件(Component)角色:定义一个抽象接口以 ...

  7. 详解设计模式:装饰器模式

    装饰器模式(Decorator Pattern)也称为包装模式(Wrapper Pattern),是 GoF 的 23 种设计模式中的一种结构型设计模式.装饰器模式 是指在不改变原有对象的基础之上,将 ...

  8. javascript设计模式之装饰器模式(结构型模式)

    javascript设计模式之装饰器模式 js的设计模式分为创建型模式,结构型模式和行为模式 结构模式描述了如何组合对象以提供新的功能. 装饰器模式是一种常见的结构型模式,我们可以以一个基础对象为基础 ...

  9. 设计模式之装饰器模式详解

    设计模式之装饰器模式详解 文章目录 设计模式之装饰器模式详解 一.什么是装饰器模式 二.装饰器模式的角色组成 三.装饰器模式通用写法示例 四.装饰器模式业务中的应用举例 五.装饰器模式优缺点 一.什么 ...

  10. 装饰者模式 php,PHP设计模式之装饰器模式

    装饰器设计模式 什么是装饰器模式 装饰器模式就是对一个已有的结构增加装饰.装饰模式是在不必改变原类文件和使用继承的情况下,动态地扩展一个对象的功能.它是通过创建一个包装对象,也就是装饰来包裹真实的对象 ...

最新文章

  1. 某单位招聘考试需要考核数学英语计算机,2010年10月自学考试电子商务网站设计原理试题...
  2. 三维等值面提取算法(Dual Contouring)
  3. Github page + octopress介绍
  4. flask的同一ip域名不同端口的两个网站session冲突
  5. hmailserver怎么搭建php,hMailServer邮件服务器安装配置(亲测可用)
  6. 横向经济联合的理论认识
  7. jsp指令元素与动作元素
  8. [自学]Docker system 命令 查看docker镜像磁盘占用情况 Docker volume 相关
  9. centos安装kvm
  10. 《MySQL——恢复数据-误删行、表、库》
  11. php解析目录函数怎么用,PHP中目录解析函数
  12. Servlet的使用你掌握多少?
  13. Struts2中的Actionsupport类
  14. 【终终极版】linux(Ubuntu)下wineQQ的安装办法
  15. mysql 敏感词_过滤敏感词方式
  16. 速领,阿里巴巴Java开发手册终极版
  17. 这种网站我看见就收藏
  18. Chrome浏览器的跨域设置----包含新老版本两种设置
  19. HTTP和RPC的区别
  20. iphone 模拟滑动_如何在iPhone或iPad上滑动输入

热门文章

  1. 新能源车融资租赁市场现状研究分析报告 -
  2. 单片机唱祝你平安c语言,蜂鸣器唱《祝你平安》51单片机
  3. python骨灰技巧_再见,Python!骨灰级程序员:用对这个方法,编程能力暴增!
  4. SLMi333国内首款兼容光耦带DESAT保护功能的隔离式栅极驱动器
  5. 计算机远程桌面删除记录,win10系统删除远程桌面历史记录的处理方法
  6. CCS使用教程01:CCS的介绍、下载与安装
  7. 南京廖华计算机二级考试答案,计算机二级MSOFFICE模拟考试题及答案题目
  8. 电脑清空回收站怎么恢复回来?(已解决)
  9. xlrd无法打开xlsx文件
  10. WiFi 6到来,WiFi也有更新换代?