php设计模式之装饰器模式

1.介绍

1.装饰器模式(Decorator),可以动态地添加修改类的功能

2.一个类提供了一项功能,如果要在修改并添加额外的功能,传统的编程模式,需要写一个子类继承它,并重新实现类的方法

3.使用装饰器模式,仅需在运行时添加一个装饰器对象即可实现,可以实现最大的灵活性。

2.实例

接下来我们举一个例子,使用php实现一个小画板的功能(画指定颜色图形)

1.没使用装饰器之前的传统方式

1.实现一个画板的类


<?php

    class Canvas
    {
        //保存点阵的一个数组
        public $data;

        //初始化点阵
        function init ( $width = 20, $height = 10 )
        {
            $data = array();
            for ($i = 0; $i < $height; $i++) {
                for ($j = 0; $j < $width; $j++) {
                    $data[ $i ][ $j ] = ‘*‘;
                }
            }
            $this->data = $data;
        }

        //初始化一个正方形的点阵
        function rect ( $a1, $a2, $b1, $b2 )
        {
            foreach ($this->data as $k1 => $line) {
                if ($k1 < $a1 or $k1 > $a2)
                    continue;
                foreach ($line as $k2 => $char) {
                    if ($k2 < $b1 or $k2 > $b2)
                        continue;
                    $this->data[ $k1 ][ $k2 ] = ‘&nbsp;&nbsp;‘;
                }
            }
        }

        //开始执行画图
        function draw ()
        {
            foreach ($this->data as $line) {
                foreach ($line as $char) {
                    echo $char;
                }
                echo "<br />\n";
            }
        }
    }

2.调用

$canvas = new Canvas();
$canvas->init(40, 20);

$canvas->rect(4,15,9,30);
$canvas->draw();

3.结果 我们看到如下一个正方形

****************************************
****************************************
****************************************
****************************************
****************************************
*********                      *********
*********                      *********
*********                      *********
*********                      *********
*********                      *********
*********                      *********
*********                      *********
*********                      *********
*********                      *********
*********                      *********
****************************************
****************************************
****************************************
****************************************
****************************************

4.下面问题来了

  1. 如果我想给这个图形添加自己想要的颜色怎么添加?

5.解决以上的问题(在显示的时候输出html代码,修改draw方法你就可以看到你想要的结果了)

        //开始执行画图
        function draw ()
        {
            echo "<div style=‘color:red;‘>";
            foreach ($this->data as $line) {
                foreach ($line as $char) {
                    echo $char;
                }
                echo "<br />\n";
            }
            echo "</div>";
        }

6.问题又来了

1.这样写硬编码了,如果那天我不想加颜色了,我想加粗,倾斜,那就需要修改代码,或者我想在上面添加一个标题。。。等等需求,下面我们就用装饰器模式来修改上面的代码,使上面的代码解耦!

2.使用装饰器模式实现上面的功能

1.实现一个装饰器的基类

//画图装饰器
interface DrawDecorator
{
    //画之前的操作
    function beforeDraw();

    //画之后的操作
    function afterDraw();
}

2.实现一个颜色装饰器

class ColorDrawDecorator implements DrawDecorator
{
    //颜色属性
    protected $color;

    //初始化颜色
    function __construct($color = ‘red‘)
    {
        $this->color = $color;
    }

    //画之前的操作
    function beforeDraw()
    {
        echo "<div style=‘color: {$this->color};‘>";
    }

    //画之后的操作
    function afterDraw()
    {
        echo "</div>";
    }
}

3.从新实现画板的类


    class Canvas
    {
        //保存点阵的一个数组
        public $data;

        //保存装饰器对象
        protected $decorators = array();

        //初始化点阵
        function init($width = 20, $height = 10)
        {
            $data = array();
            for($i = 0; $i < $height; $i++)
            {
                for($j = 0; $j < $width; $j++)
                {
                    $data[$i][$j] = ‘*‘;
                }
            }
            $this->data = $data;
        }

        //注册装饰器对象
        function addDecorator(DrawDecorator $decorator)
        {
            $this->decorators[] = $decorator;
        }

        //画之前的操作
        function beforeDraw()
        {
            foreach($this->decorators as $decorator)
            {
                $decorator->beforeDraw();
            }
        }

        //画之后的操作
        function afterDraw()
        {
            $decorators = array_reverse($this->decorators);
            foreach($decorators as $decorator)
            {
                $decorator->afterDraw();
            }
        }

        //开始画图
        function draw()
        {
            $this->beforeDraw();
            foreach($this->data as $line)
            {
                foreach($line as $char)
                {
                    echo $char;
                }
                echo "<br />\n";
            }
            $this->afterDraw();
        }

        //描述一个矩形的点阵
        function rect($a1, $a2, $b1, $b2)
        {
            foreach($this->data as $k1 => $line)
            {
                if ($k1 < $a1 or $k1 > $a2) continue;
                foreach($line as $k2 => $char)
                {
                    if ($k2 < $b1 or $k2 > $b2) continue;
                    $this->data[$k1][$k2] = ‘&nbsp;‘;
                }
            }
        }
    }

4.调用

$canvas = new Canvas();
//注入装饰器对象
$canvas->addDecorator(new ColorDrawDecorator(‘green‘));
$canvas->init(40, 20);

$canvas->rect(4,15,9,30);
$canvas->draw();

5.结果

输出一个绿色的矩形

同样如果你还想使用加粗,倾斜,设置自定义标题等等,就在创建一个特定的装饰器,注入到画布内就可以实现了

6.总结

1.装饰器就是在执行特定操作之前,加入你自定义的一些操作

2.装饰器的实现,好比钩子(hook)的机制, 比如drupal中的hook机制

3.使用call_user_func 或者 call_user_func_array 也可实现该装饰器机制,这个可以参考drupal的hook实现,也挺不错的!这里先不介绍了,不在设计模式之内,回头有时间在写一下。

版权声明:本文为博主原创文章,未经博主允许不得转载。

时间: 2024-10-13 21:18:37

php设计模式之装饰器模式的相关文章

java设计模式之 装饰器模式

适AT java设计模式之 装饰器模式 装饰器模式 装饰器模式(Decorator Pattern)允许向一个现有的对象添加新的功能,同时又不改变其结构. 这种类型的设计模式属于结构型模式,它是作为现有的类的一个包装. 这种模式创建了一个装饰类,用来包装原有的类,并在保持类方法签名完整性的前提下,动态给一个对象添提供了额外的功能. 我们通过下面的实例来演示装饰器模式的用法.模拟一个人从想吃饭.找饭店.享受美食.结束吃饭的过程 代码展示: 首先创建一个被修饰的接口 Eat package deco

python设计模式之装饰器模式

装饰器模式 装饰器模式(Decorator Pattern)允许向一个现有的对象添加新的功能,同时又不改变其结构.这种类型的设计模式属于结构型模式,它是作为现有的类的一个包装. 这种模式创建了一个装饰类,用来包装原有的类,并在保持类方法签名完整性的前提下,提供了额外的功能. import time def log_calls(func): def wrapper(*args,**kwargs): now=time.time() print("Calling{0} with {1} and {2}

设计模式之: 装饰器模式

什么是装饰器模式 作为一种结构型模式, 装饰器(Decorator)模式就是对一个已有结构增加"装饰". 适配器模式, 是为现在有结构增加的是一个适配器类,.将一个类的接口,转换成客户期望的另外一个接口.适配器让原本接口不兼容的类可以很好的合作. 装饰器模式是将一个对象包装起来以增强新的行为和责任.装饰器也称为包装器(类似于适配器) 有些设计设计模式包含一个抽象类,而且该抽象类还继承了另一个抽象类,这种设计模式为数不多,而装饰器就是其中之一. 什么时候使用装饰器模式 基本说来, 如果想

设计模式之装饰器模式

一.百科 概述: 23种设计模式之一,英文叫Decorator Pattern,又叫装饰者模式.装饰模式是在不必改变原类文件和使用继承的情况下,动态地扩展一个对象的功能.它是通过创建一个包装对象,也就是装饰来包裹真实的对象. 特点: (1) 装饰对象和真实对象有相同的接口.这样客户端对象就能以和真实对象相同的方式和装饰对象交互. (2) 装饰对象包含一个真实对象的引用(reference) (3) 装饰对象接受所有来自客户端的请求.它把这些请求转发给真实的对象. (4) 装饰对象可以在转发这些请

五分钟一个设计模式之装饰器模式

五分钟一个设计模式,用最简单的方法来描述设计模式.查看更多设计模式,请点击五分钟一个设计模式系列 http://blog.csdn.net/daguanjia11/article/category/3259443 示例代码 今天实在是想不出什么开场白来引出示例了,也想不出特别有意思的示例了,就用一个很土的例子来描述一下装饰器模式吧. 每个人都要穿衣服,每件衣服都具有某些特定的功能,例如,穿上鞋子的话走路舒服,穿上上衣的话上身不冷,穿上裤子的话腿不冷(你这不废话吗).也就是说,每件衣服都具有特定的

23种设计模式之装饰器模式(Decorator Pattern)

装饰器模式(Decorator Pattern) 允许向一个现有的对象添加新的功能,同时又不改变其结构.这种类型的设计模式属于结构型模式,它是作为现有的类的一个包装. 这种模式创建了一个装饰类,用来包装原有的类,并在保持类方法签名完整性的前提下,提供了额外的功能. 通过采用组合.而非继承的手法,Decorator模式实现了在运行时动态地扩展对象功能的能力,而且可以根据需要扩展多个功能.避免了单独使用继承带来的“灵活性差"和"多子类衍生问题". 优点:装饰类和被装饰类可以独立发

浅谈设计模式(二):装饰器模式|中介模式|原型模式

装饰器模式(Decorator Pattern) 装饰器模式可用来给一个类动态添加功能,将其装饰成一个新的类.这就是装饰器的概念.看到这里我们可能会想,要达到这种效果,我们用子类继承父类不就可以了吗? 没错装饰器模式,本身是一种继承的替代方案.那既然是替代方案,那么自然就有它不一样的地方. 具体区别在哪里呢? 请看 装饰器模式更灵活:继承时父子类的关系是静态的,而装饰器模式是动态的,装饰类和被装饰类的关系是运行时候确认的 装饰类和被装饰类的耦合关系是松散的,各自可以独立变化 下面看看具体的代码.

设计模式之装饰器模式io的小入门(十一)

装饰器模式详解地址 原文总结 定义: 在不必改变原类文件和使用继承的情况下, 动态的扩展一个对象的功能. 通过创建一个包装对象, 也就是装饰来包裹真实的对象 部分详解提示 看了一些文档, 装饰器模式非常依赖构造器 与 重写方法 装饰器模式的特点: 不改变原来的类 , 不使用继承 , 动态扩展 流这块除了文件上传下载使用过, 确实用的太少了这里继续复习下最简单的文件上传 文件目录的创建 目录的是否存在没有就创建 不推荐: 年/月/日 一般没什么用户权限的图片之类的不过 推荐: 模块/用户/模块/年

java设计模式之装饰器模式

装饰器模式的应用场景:1.需要扩展一个类的功能.2.动态的为一个对象增加功能,而且还能动态撤销.(继承不能做到这一点,继承的功能是静态的,不能动态增删.) 源接口: 1 public interface Sourceable { 2 3 void method(); 4 } Source类: 1 public class Source implements Sourceable { 2 @Override 3 public void method() { 4 System.out.println