php设计模式(1)-- 观察者模式 -- spl标准写法
php设计模式(2)-- 观察者模式 -- 用trait来改进的写法
php设计模式(3)-- 责任链(职责链)模式
php设计模式(4)-- 装饰器模式
分析
网上的套话就不说了。
图片来自红黑联盟:
上图中,Componet 对应我这里的 Display
ConcreteComponet 对应我这里的 BasicDisplay
Decorator 对应我这里的 Border
剩下两个分别对应 FullBorder 和 SiderBorder
装饰器适用场合:假设一个对象有某种功能,在有些时候,需要对这功能增强,但有些时候,又需要使用原有功能。这时使用装饰器。
装饰器和责任链都会有一个长长的对象关联链条,其差异是:责任链强调对同一个消息的不同处理,而不是功能增强。 责任链是多个功能,而 装饰器是一个功能,但是这个功能有时需要一些小改变,改来改去是同一个功能。
java的流使用了装饰器,说明装饰器也可以改变结果的表现形式,总之,功能增强是个泛泛而言的事情,能做到事情很多。
注意:和观察者,以及责任链一样,类与类的具体哪个和哪个关联,先后顺序等,都放在客户端代码里!
现在我们构造需求。
假设,我们有一个文档,需要打印机打印出来,但是我们希望打印一些边框,让文档漂亮些,边框跟文档内容没关系。但是同一个功能:打印。所以我们使用装饰器。
代码实现
<?php /** * 装饰器模式学习代码。 * * 位于最顶层,表示了整个设计模式示例的功能:打印字符串 * * 这个程序也可以包装多行的文本,只是代码改得复杂一些,不利于看清设计模式。 */ abstract class Display { public abstract function getColumns(); //取得横向的字数,把责任委托给子类,所以抽象,下同 //观察子类可知,只要有一个类使用到了, //需要所有的类都要有这个方法! public abstract function getRows(); //取得纵向的行数,把责任委托给子类 public abstract function getRowText($row);//取得第row行的字符串 public function show() { //因为这个方法的实现是固定的,所以写这里 for ($i = 0; $i < $this->getRows(); $i++) { echo $this->getRowText($i) . PHP_EOL; } echo PHP_EOL; } } /** * 注意此类一定被包裹在核心,和别的类不同,虽然都是继承Display类 * 所以我取名basic */ class BasicDisplay extends Display { private $string; //最里面的,一定会被打印出的字符串 public function __construct($string) { //需要在外部指定 $this->string = $string; } public function getColumns() { //注意!,仅被某类调用,却写到每个类中!! return strlen($this->string); } public function getRows() { //核心只打印一行 return 1; } public function getRowText($row) { //仅在row为0时才返回 if ($row == 0) { return $this->string; } else { return null; } } } /** * 因为外框又有多种,所以把共性抽取出来,形成此抽象类,其中, * 还确定了每个装饰器子类都有的构造方法和属性,通常就是属于共同接口的对象 */ abstract class Border extends Display { protected $display; //注意到:是同一接口的对象,php //不像java能表达出类型,但实际是的 protected function __construct(Display $display) { //后面可看到,子类实际可以扩展构造方法 $this->display = $display; } } /** * 在字符两边输出特定字符(由程序外部指定)的外框类, * 通过Border间接继承Display * */ class SideBorder extends Border { //装饰用的字符,会写到两边 private $borderChar; public function __construct(Display $display, $ch) {//注意重写了构造方法。 parent::__construct($display); $this->borderChar = $ch; } public function getColumns() {// 左右各加一个字符,所以宽度加2 return 1+ $this->display->getColumns() + 1; } public function getRows() { return $this->display->getRows(); } /** * 最后的显示效果如 |hello, world| * 其中两边的|只是示例,由外部传入的。 * 根据php的类型,没有字符类,所以请确保只传入一个字符。这里没有判断,也可以抛异常等。 */ public function getRowText($row) { // 注意这其实在一个循环里,只是每行做同样的处理罢了。 return $this->borderChar . $this->display->getRowText($row) . $this->borderChar; } } /** * 把字符包裹于其中的外框类 * 通过Border间接继承Display * */ class FullBorder extends Border { private $borderChar; public function __construct(Display $display) { parent::__construct($display); } //这些方法很重要,保证了上下的字符对齐(假定字符宽度相等) //注意到:虽然别的类的该方法似乎没有用到, //实际在这里用到了,让本类可以知道里面内核的字符宽度 public function getColumns() { return 1 + $this->display->getColumns() + 1; } public function getRows() { return 1 + $this->display->getRows() + 1; } /** * 把行数确定为核心内容加2后,见上getRows,就可以在顶部和底部输出装饰 * +-------------------+ * +-------------------+ * * 然后,在内容的两边输出 | 字符 */ public function getRowText($row) { if ($row == 0) { // 第1行 return '+' . $this->makeLine('-', $this->display->getColumns()) . '+'; } elseif ($row == $this->display->getRows() + 1) { // 最后一行,= 原有总行数 + 1,因为行数从1算,row从0算。 return '+' . $this->makeLine('-', $this->display->getColumns()) . '+'; } else { return '|' . $this->display->getRowText($row - 1) . '|';//-1 是因为有一个错位,多了一行 } } private function makeLine($ch, $count) { $s = ''; for ($i = 0; $i < $count; $i++) { $s .= $ch; } return $s; } } //打印“Hello,world”,没有任何装饰 $b1 = new BasicDisplay('Hello, world.'); $b1->show(); //把装饰字符'#'加在b1的左右两边 $b2 = new SideBorder($b1, '#'); $b2->show(); //把b2加上装饰外框 $b3 = new FullBorder($b2); $b3->show(); //b4在核心的外面加上了多重外框,请仔细观察图形与每个装饰器的对应关系,很有意思的。 $b4 = new SideBorder( new FullBorder( new FullBorder( new SideBorder( new FullBorder( new BasicDisplay('Hello, world.') ), '*' ) ) ), '/' ); $b4->show();
结果展示,非常精巧
Hello, world. #Hello, world.# +---------------+ |#Hello, world.#| +---------------+ /+-------------------+/ /|+-----------------+|/ /||*+-------------+*||/ /||*|Hello, world.|*||/ /||*+-------------+*||/ /|+-----------------+|/ /+-------------------+/