GoF定义:动态地给一个工具添加一些额外的职责,就增加功能来说,Decorator模式比较天生子类更为灵巧
GoF类图
装饰器方法构造类图
代码实现
interfaceComponent{publicfunctionoperation();}classConcreteComponentimplementsComponent{publicfunctionoperation(){echo"I'mface!".PHP_EOL;}}
很大略的一个接口和一个实现,这里我们就把详细的实现类看作是一张脸吧!
abstractclassDecoratorimplementsComponent{protected$component;publicfunction__construct(Component$component){$this->component=$component;}}
抽象的装饰者类,实现Component接口,但并不实现operation()方法,让子类去实现。在这里紧张保存一个Componet的引用,一会就要对他进行装饰。对应到上方的详细类,我们便是要准备给脸扮装啦!
classConcreteDecoratorAextendsDecorator{public$addedState=1;//没什么实际意义的属性,只是差异于ConcreteDecoratorBpublicfunctionoperation(){echo$this->component->operation()."Push".$this->addedState."cream!
".PHP_EOL;}}classConcreteDecoratorBextendsDecorator{publicfunctionoperation(){$this->component->operation();$this->addedBehavior();}//没什么实际意义的方法,只是差异于ConcreteDecoratorApublicfunctionaddedBehavior(){echo"Push2cream!
".PHP_EOL;}}
两个详细装饰者。在这里我是涂了两次霜,毕竟是纯爷们,对扮装这事儿真的是不理解。彷佛第一步该当先是打粉底吧?不过这次就这样,我们这两个装饰器实现的便是给脸上涂两层霜。
从代码中可以看出,我们是一贯对详细的那个ConcreteComponent工具来进行包装再往下的话实在我们是对他的operation()这个方法包装了两次,每次都是在前一次的根本上加了一点点东西不要纠结于A和B装饰器上的added属性和方法,他们只是GoF类图中用以差异这两个装饰器不是同一个东西,每个装饰器都可以干很多别的事,Component工具也不一定只有operation()这一个方法,我们可以选择性的去装饰工具中的全部或者部分方法彷佛我们都继续了Component,直接子类一起重写不就行了,搞这费劲干嘛?亲,理解下组合的观点哟,我们的Decorator父类里面是一个真实工具的引用哦,解耦了自身哦,我们只给真实的工具去做包装,您可别直接实例化妆饰器来直接用还是没懂?好处呢?老系统的类啊、方法啊你敢随便乱改?想给前任写的牛(S)逼(B)代码扩展新功能时不妨试试装饰器这货,说不定有奇效!手机这玩意干不过某米、某O、某为,这没法玩呀,好吧,哥们去专心做手机壳吧!
嗯,我先准备了一个透明壳(Component),貌似有点丑,没办法,谁叫哥们穷。给某米的加上各种纯色(DecoratorA1),然后背后印上各种颜色的植物(DecoratorB1)吧;某O的手机最近喜好找流量明显做代言,那我给他的手机壳就用各种炫彩色(DecoratorA2)和明星的卡通头像(DecoratorB2);末了的某为,彷佛手机已经开始引领业界潮流了,折叠屏这玩意不是要砸我这卖手机壳的买卖嘛!
!
好吧,哥不给你们做了,还是跟我的某米、某O混去吧!
!
完全代码:装饰器模式
实例连续来发短信,之前我们用工厂模式办理了多个短信运营商的问题。这回我们要办理的是短信内容模板的问题。对付推广类的短信来说,根据最新的广告法,我们是不能涌现“全国第一”、“全天下第一”这类的词语的,当然,一些不太文明的用语我们也是不能利用的。
现在的情形是这样的,我们有一个很早之前的短信模板类,里面的内容是固定的,老系统依然还是利用这个模板,老系统是面对的内部员工,对措辞内容的哀求不高。而新系统则须要向全网发送,也便是内外部的用户都要发送。这时,我们可以用装饰器模式来对老系统的短信模板进行包装。实在说大略点,我们便是用装饰器来做文本更换的功能。好处呢?当然是可以不去改动原来的模板类中的方法就实现了对老模板内容的修正扩展等。
短信发送类图
短信发送装饰器方法
完全源码:短信发送装饰器方法
<?php//短信模板接口interfaceMessageTemplate{publicfunctionmessage();}//假设有很多模板实现了上面的短信模板接口//下面这个是个中一个优惠券发送的模板实现classCouponMessageTemplateimplementsMessageTemplate{publicfunctionmessage(){return'优惠券信息:我们是全国第一的牛X产品哦,送您十张优惠券!
';}}//我们来准备好装饰上面那个过期的短信模板abstractclassDecoratorMessageTemplateimplementsMessageTemplate{public$template;publicfunction__construct($template){$this->template=$template;}}//过滤新广告法中不许可涌现的词汇classAdFilterDecoratorMessageextendsDecoratorMessageTemplate{publicfunctionmessage(){returnstr_replace('全国第一','全国第二',$this->template->message());}}//利用我们的大数据部门同本家儿动天生的新词库来过滤敏感词汇,这块过滤不是逼迫要过滤的内容,可选择利用classSensitiveFilterDecoratorMessageextendsDecoratorMessageTemplate{public$bigDataFilterWords=['牛X'];public$bigDataReplaceWords=['好用'];publicfunctionmessage(){returnstr_replace($this->bigDataFilterWords,$this->bigDataReplaceWords,$this->template->message());}}//客户端,发送接口,须要利用模板来进行短信发送classMessage{public$msgType='old';publicfunctionsend(MessageTemplate$mt){//发送出去咯if($this->msgType=='old'){echo'面向内网用户发送'.$mt->message().PHP_EOL;}elseif($this->msgType=='new'){echo'面向全网用户发送'.$mt->message().PHP_EOL;}}}$template=newCouponMessageTemplate();$message=newMessage();//老系统,用不着过滤,只有内部用户才看得到$message->send($template);//新系统,面向全网发布的,须要过滤一下内容哦$message->msgType='new';$template=newAdFilterDecoratorMessage($template);$template=newSensitiveFilterDecoratorMessage($template);//过滤完了,发送吧$message->send($template);
解释
装饰器的最大好处:一是不改变原有代码的情形下对原有代码中的内容进行扩展,开放封闭原则;二是每个装饰器完本钱身的功能,单一职责;三是用组合实现了继续的觉得;最适用于:给老系统进行扩展要小心:过多的装饰者会把你搞晕的不一定都是对同一个方法进行装饰,实在装饰者该当更多的用于对工具的装饰,对工具进行扩展,这里我们都是针对一个方法的输出进行装饰,但仅限此文,装饰器的运用实在更加广泛装饰器的特点是全部都继续自一个主接口或类,这样的好处便是返回的工具是相同的抽象数据,具有相同的行为属性,否则,就不是装饰之前的工具,而是一个新工具了有点不好理解没紧要,我们这次的例子实在也很勉强,这个设计模式在《Head First设计模式》中有提到Java的I/O系列接口是利用的这种设计模式:FileInputStream、LineNumberInputStream、BufferInputStream等Laravel框架中的中间件管道,这里实在是多种模式的综合运用,个中也运用到了装饰器模式:Laravel HTTP——Pipeline 中间件装饰者模式源码剖析其余在Laravel中,日志处理这里也是对Monolog进行了装饰,有兴趣的同学可以去理解下下期看点又是大伽驾到,电源适配器理解吧?变压器总见过吧?你可能用过,也可能没用过,但你一定听说过这个非常非常出名的适配器模式。