6.4装饰模式

装饰模式(Decorator),动态地给一个对象添加一些额外的职责,就增加功能来说,装饰模式比生成子类更为灵活。 第六章 穿什么有这么重要?——装饰模式 - 图1Component是定义一个对象接口,可以给这些对象动态地添加职责。ConcreteComponent是定义了一个具体的对象,也可以给这个对象添加一些职责。Decorator,装饰抽象类,继承了Component,从外类来扩展Component类的功能,但对于Component来说,是无需知道Decorator的存在的。至于ConcreteDecorator就是具体的装饰对象,起到给Component添加职责的功能。
代码示例:
Component类

  1. abstract class Component
  2. {
  3. abstract function Operation();
  4. }

ConcreteComponent类

  1. class ConcreteComponent extends Component
  2. {
  3. public function Operation()
  4. {
  5. echo '具体对象的操作';
  6. }
  7. }

Decorator类

  1. abstract class Decorator extends Component
  2. {
  3. protected $component = null;
  4. public function setComponent($component) //设置Component
  5. {
  6. $this->component = $component;
  7. }
  8. //重写Operation(),实际执行的是Component的Operation()
  9. public function Operation()
  10. {
  11. if ($this->component != null) {
  12. $this->component->Operation();
  13. }
  14. }
  15. }

ConcreteDecorator 装饰类

  1. class ConcreteDecoratorA extends Decorator
  2. {
  3. //本类独有功能,以区别于ConcreteDecoratorB
  4. private $addedState = '';
  5. public function Operation()
  6. {
  7. //首先运行原Component的Operation(),再执行
  8. //本类的功能,如addedState,相当于对原Component
  9. //进行了装饰
  10. parent::Operation();
  11. $this->addedState = 'New State';
  12. echo 'A类独有参数:addedState => ' . $this->addedState . '<br>';
  13. echo '具体装饰对象A的操作<br>';
  14. }
  15. }
  16. class ConcreteDecoratorB extends Decorator
  17. {
  18. public function Operation()
  19. {
  20. //首先运行原Component的Operation(),再执行
  21. //本类的功能,如AddedBehavior(),相当于对原
  22. //Component进行了装饰
  23. parent::Operation();
  24. $this->AddedBehavior();
  25. echo '具体装饰对象B的操作<br>';
  26. }
  27. //本类独有的方法,以区别于ConcreteDecoratorA
  28. private function AddedBehavior()
  29. {
  30. echo 'B类独有方法AddedBehavior<br>';
  31. }
  32. }

客户端代码

  1. //装饰的方法是:首先用ConcreteComponent实例
  2. //化对象c,然后用ConcreteDecoratorA的实例化
  3. //对象d1来包装c,再用ConcreteDecoratorB 的
  4. //对象d2包装d1,最终执行d2的Operation()
  5. $c = new ConcreteComponent();
  6. $d1 = new ConcreteDecoratorA();
  7. $d2 = new ConcreteDecoratorB();
  8. $d1->setComponent($c);
  9. $d2->setComponent($d1);
  10. $d2->Operation();

装饰模式是利用SetComponent来对对象进行包装的。这样每个装饰对象的实现就和如何使用这个对象分离开了,每个装饰对象只关心自己的功能,不需要关心如何被添加到对象链当中。
如果只有一个ConcreteComponent类而没有抽象的Component类,那么Decorator类可以是ConcreteComponent的一个子类。同样道理,如果只有一个ConcreteDecorator类,那么就没有必要建立一个单独的Decorator类,而可以把Decorator和ConcreteDecorator的责任合并成一个类。

6.5小菜扮靓第三版

代码结构图: 第六章 穿什么有这么重要?——装饰模式 - 图2代码实现:
“Person” 类(ConcreteComponent)

  1. class Person
  2. {
  3. private $name = '';
  4. public function __construct($name)
  5. {
  6. $this->name = $name;
  7. }
  8. public function show()
  9. {
  10. echo "装扮的" . $this->name . '<br>';
  11. }
  12. }

服饰类(Decorator)

  1. class Finery
  2. {
  3. protected $component = null;
  4. //打扮
  5. public function decorate($component)
  6. {
  7. $this->component = $component;
  8. }
  9. public function show()
  10. {
  11. if ($this->component != null) {
  12. $this->component->show();
  13. }
  14. }
  15. }

具体 服饰类(ConcreteDecorator)

  1. class TShirts extends Finery
  2. {
  3. public function show()
  4. {
  5. echo '大T恤 ';
  6. parent::show();
  7. }
  8. }
  9. class BigTrouser extends Finery
  10. {
  11. public function show()
  12. {
  13. echo '垮裤 ';
  14. parent::show();
  15. }
  16. }
  17. class Panties extends Finery
  18. {
  19. public function show()
  20. {
  21. echo '裤头 ';
  22. parent::show();
  23. }
  24. }

客户端代码

  1. $xc = new Person("小菜");
  2. echo '第一种装扮:<br>';
  3. $ts = new TShirts();
  4. $bt = new BigTrouser();
  5. //装饰过程
  6. $ts->decorate($xc);
  7. $bt->decorate($ts);
  8. $bt->show();
  9. echo '第二种装扮:<br>';
  10. //装饰过程
  11. $pan = new Panties();
  12. $pan->decorate($bt);
  13. $pan->show();

6.6装饰模式总结

装饰模式是为已有功能动态地添加更多功能的一种方式。
装饰模式的使用场景:
在起初的设计中,当系统需要新功能的时候,是向旧的类中添加新的代码。这些新加的代码通常装饰了原有类的核心职责或主要行为,比如用西装或嘻哈服来装饰小菜,但这种做法的问题在于,它们在主类中加入了新的字段,新的方法和新的逻辑,从而增加了主类的复杂度,就像你起初的那个‘人’类,而这些新加入的东西仅仅是为了满足一些只在某种特定情况下才会执行的特殊行为的需要。装饰模式却提供了一个非常好的解决方案,它把每个要装饰的功能放在单独的类中,并让这个类包装它所要装饰的对象,因此,当需要执行特殊行为时,客户代码就可以在运行时根据需要有选择地、按顺序地使用装饰功能包装对象了。
装饰模式的优点:

  • 把类中的装饰功能从类中搬移去除,这样可以简化原有的类。
  • 有效地把类地核心职责和装饰功能区分开。
  • 而且可以去除相关类中重复的装饰逻辑。

注意:装饰模式的装饰顺序很重要,比如加密数据和过滤词汇都可以是数据持久化前的装饰功能,但若先加密了数据再用过滤功能就会出问题了,最理想的情况,是保证装饰类之间彼此独立,这样它们就可以以任意的顺序进行组合了。