装饰器模式

问题:编程人员最害怕听到的一句话是:“这是一个不断变化的需求文档”。客户在指定需求收集和规范创建阶段自始至终都需要进行持续开发。即使在这两个阶段完成之后,客户还可能返回上述阶段并要求在很多地方进行一些调整。因为这些修改对于客户来说似乎非常细小,所以他们并不期望在部署时再进行优化处理。如果没有进行正确的处理,那么即使是很小的调整也会让人头疼。如果要对基本功能稍加改动,那么最好选择使用装饰器设计模式。

介绍:装饰器模式就是对一个已有的结构增加“装饰”,从一个对象外部来给对象添加功能,同时又不改变其结构。常给对象添加功能,要么直接修改对象添加相应的功能,要么派生对应的子类来扩展,抑或是使用对象组合的方式。显然,直接修改对应的类这种方式并不可取。在面向对象的设计中,而我们也应该尽量使用对象组合,而不是对象继承来扩展和复用功能。装饰器模式就是基于对象组合的方式,可以很灵活的给对象添加所需要的功能。装饰器模式的本质就是动态组合。动态是手段,组合才是目的。总之,装饰模式是通过把复杂的功能简单化,分散化,然后再运行期间,根据需要来动态组合的这样一个模式。装饰器和适配器都属于结构性设计模式,对于适配器模式,为现有结构增加的是一个适配器类,用来处理不兼容的接口。装饰器模式会向现有对象增加对象。

何时使用:如果已有对象的部分内容或功能性发生改变,但是不需要修改原始对象的结构。

装饰器模式结构图


decoration.png

简单事例代码:

  我们假设某个弓箭手使用普通弓箭的伤害值为300,而嵌入了某种属性的物质之后会增加攻击力,那么,此时,我们就相当于用物质来装饰原有的那个武器。


//武器接口
abstract class IWeapon{
 abstract public function getName();
 abstract public function setName($Arm_name);
  abstract public function getAttack();
}


//角色接口
abstract class IRole{
  abstract public function getName();
  abstract public function setName($Role_name);
  abstract public function getAttack(IWeapon $Weapon);
}

//创建武器

class Arm extends IWeapon{
  private $name;
  public function setName($Arm_name){
    $this->name = $Arm_name;
  }
  
  public function getName(){
    return $this->name;
  }
  
  public function getAttack(){
    return 300;
  }
}


//创建角色

class Role extends IRole{
  private $Role;
  public function setName($Role_name){
    $this->Role = $Role_name;
  }
  public function getName(){
    return $this->Role_name;
  }
  public function getAttack(IWeapon $weapon){
    return $this->Role . '使用' . $weapon->getName() . '进行攻击,造成的伤害为:' . $weapon->getAttack();
  }
}

下面我们来使用给武器进行伤害加成的装饰


abstract class Decorator extends IWeapon{
    //抽象类继承抽象类时,子抽象类可以不实现父抽象类的所有方法
    //这里不实现任何一个抽象方法,只是维护Weapon的引用
}


/攻击加成类

class nucleus extends Decorator{
  private $name;
  private $yuanxian_weapon;
  
  public function __construct(IWeapon $weapon){
     $this->yuanxian_weapon = $weapon;
  }
  
  public function setName($Arm_name){
   return $this->name = $Arm_name;
  }
  
  public function getName(){
   return '嵌套了' . $this->name . '的' . $this->yuanxian_weapon->getName();
  }
  
  public function getAttack(){
   return $this->yuanxian_weapon->getAttack()+10;
  }
}


class client{
  public function text(){
    $role = new Role();
    $role->setName('弓箭手');
    $arm = new Arm();
    $arm->setName('弓箭');
    echo $role->getAttack($arm),'<br />';
    
    $add = new nucleus($arm);
    $add->setName('一阶魔核');
    echo $role->getAttack($add);
  }
}

$work = new client();
$work->text();


最终的结果为:

decorator_result.JPG



头像
QQ登录: