工厂模式告一段落,咱们来研究其余一些模式。不知道各位大佬有没有尝试过女装?听说女装大佬程序员不少哟。其实,今天的装饰器模式就和化妆这件事很像。相信若是有程序媛MM在的话,立刻就能和你讲清楚这个设计模式。php
装饰这两个字,咱们暂且把他变成化妆。首先你得有一张脸,而后打底,而后上妆,能够早上来个淡妆上班,也能够下班的时候补成浓妆出去嗨。固然,码农们下班的时间点正好是能遇上夜场的下半场的。话说回来,无论怎么化妆,你的脸仍是你的脸,有可能能够化成别人不认识的另外一我的,但这的的确确仍是你的脸。这就是装饰器,对对象(脸)进行各类装饰(化妆),让这个脸更好看(增长职责)。laravel
GoF定义:动态地给一个对象添加一些额外的职责,就增长功能来讲,Decorator模式相比生成子类更为灵活git
GoF类图程序员
代码实现github
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; // 没什么实际意义的属性,只是区别于ConcreteDecoratorB
public function operation(){
echo $this->component->operation() . "Push " . $this->addedState . " cream!" . PHP_EOL;
}
}
class ConcreteDecoratorB extends Decorator{
public function operation(){
$this->component->operation();
$this->addedBehavior();
}
// 没什么实际意义的方法,只是区别于ConcreteDecoratorA
public function addedBehavior(){
echo "Push 2 cream!" . PHP_EOL;
}
}
复制代码
两个具体装饰者。在这里我是涂了两次霜,毕竟是纯爷们,对化妆这事儿真的是不了解。好像第一步应该先是打粉底吧?不过此次就这样,咱们这两个装饰器实现的就是给脸上涂两层霜。源码分析
手机这玩意干不过某米、某O、某为,这无法玩呀,好吧,哥们去专心作手机壳吧!嗯,我先准备了一个透明壳(Component),貌似有点丑,没办法,谁叫哥们穷。给某米的加上各类纯色(DecoratorA1),而后背后印上各类颜色的植物(DecoratorB1)吧;某O的手机最近喜欢找流量明显作代言,那我给他的手机壳就用各类炫彩色(DecoratorA2)和明星的卡通头像(DecoratorB2);最后的某为,好像手机已经开始引领业界潮流了,折叠屏这玩意不是要砸我这卖手机壳的生意嘛!!好吧,哥不给大家作了,仍是跟个人某米、某O混去吧!!大数据
完整代码:装饰器模式this
继续来发短信,以前咱们用工厂模式解决了多个短信运营商的问题。这回咱们要解决的是短信内容模板的问题。对于推广类的短信来讲,根据最新的广告法,咱们是不能出现“全国第一”、“全世界第一”这类的词语的,固然,一些不太文明的用语咱们也是不能使用的。
如今的状况是这样的,咱们有一个很早以前的短信模板类,里面的内容是固定的,老系统依然仍是使用这个模板,老系统是面对的内部员工,对语言内容的要求不高。而新系统则须要向全网发送,也就是内外部的用户都要发送。这时,咱们能够用装饰器模式来对老系统的短信模板进行包装。其实说简单点,咱们就是用装饰器来作文本替换的功能。好处呢?固然是能够不去改动原来的模板类中的方法就实现了对老模板内容的修改扩展等。
短信发送类图
完整源码:短信发送装饰器方法
<?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);
复制代码
说明
又是大伽驾到,电源适配器了解吧?变压器总见过吧?你可能用过,也可能没用过,但你必定据说过这个很是很是出名的适配器模式。