PHP设计模式之装饰器模式

PHP设计模式之装饰器模式

工厂模式告一段落,咱们来研究其余一些模式。不知道各位大佬有没有尝试过女装?听说女装大佬程序员不少哟。其实,今天的装饰器模式就和化妆这件事很像。相信若是有程序媛MM在的话,立刻就能和你讲清楚这个设计模式。php

Gof类图及解释

装饰这两个字,咱们暂且把他变成化妆。首先你得有一张脸,而后打底,而后上妆,能够早上来个淡妆上班,也能够下班的时候补成浓妆出去嗨。固然,码农们下班的时间点正好是能遇上夜场的下半场的。话说回来,无论怎么化妆,你的脸仍是你的脸,有可能能够化成别人不认识的另外一我的,但这的的确确仍是你的脸。这就是装饰器,对对象(脸)进行各类装饰(化妆),让这个脸更好看(增长职责)。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;
    }
}
复制代码

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

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

手机这玩意干不过某米、某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);

复制代码

说明

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

下期看点

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

相关文章
相关标签/搜索