[Design Pattern] The Decorator Pattern

    作者:Flyingis

    按照常理,还是先阐述基本概念,Decorator Pattern可以用即时战略游戏中兵种的装甲强度来理解,举一个典型的例子,魔兽争霸(或冰峰王座等)中山丘是个非常厉害的角色了,经常能够一锤击毙敌人的英雄和士兵,因此被誉为英雄杀手。既然是英雄杀手,时常需要冲锋陷阵,在作战过程中自然会面临敌人的围攻,此时我们有多种方式来提升山丘的抗击打能力,一种是升级他的护甲,一种是通过魔法师给他施加增加防护的魔法,一种是等级到6时使用终极魔法来大幅度提高装甲的防护,还有一种就是使用无敌的魔法瓶,在规定时间内谁都拿他没辙,等等。虽然我不知道暴雪公司的工程师具体是如何实现这种功能设计的,但绝对不会是准备多个具有不同防御等级的山丘对象来供程序调用,如Shanqiu1、Shanqiu2……ShanqiuN,这样设计笨拙,代码繁多,如果游戏中其他兵种的装甲、攻击力的设计都是如此,那即使是天才团队也会天天在上帝面前祷告的。

    Decorator Pattern在这种情况下就可以发挥作用了,在普通装甲升级时,使用普通的装甲升级的装饰器,在使用终极魔法时,使用终极魔法装甲升级的装饰器。

ExpandedBlockStart.gif ContractedBlock.gif public   abstract   class  Soldier  dot.gif {
InBlock.gif  String info 
= "Soldier's name:";
ExpandedSubBlockStart.gifContractedSubBlock.gif  
public String getInfo() dot.gif{
InBlock.gif    
return info;
ExpandedSubBlockEnd.gif  }

InBlock.gif  
public abstract int armor();
ExpandedBlockEnd.gif}

None.gif
ExpandedBlockStart.gifContractedBlock.gif
public   abstract   class  SoldierDecorator extends Soldier dot.gif {
InBlock.gif  
public abstract String getInfo();
ExpandedBlockEnd.gif}

None.gif
ExpandedBlockStart.gifContractedBlock.gif
public   class  NormalArmorUpdate  extends  SoldierDecorator  dot.gif {
InBlock.gif  Soldier soldier;
ExpandedSubBlockStart.gifContractedSubBlock.gif  
public NormalArmorUpdate(Soldier soldier) dot.gif{
InBlock.gif    
this.soldier = soldier;
ExpandedSubBlockEnd.gif  }

InBlock.gif
ExpandedSubBlockStart.gifContractedSubBlock.gif  
public String getInfo() dot.gif{
InBlock.gif    
return soldier.getInfo() + "Shanqiu normal update!";
ExpandedSubBlockEnd.gif  }

InBlock.gif
ExpandedSubBlockStart.gifContractedSubBlock.gif  
public int armor() dot.gif{
InBlock.gif    
return soldier.armor() + 1;
ExpandedSubBlockEnd.gif  }

ExpandedBlockEnd.gif}

None.gif
ExpandedBlockStart.gifContractedBlock.gif
public   class  MagicArmorUpdate  extends  SoldierDecorator  dot.gif {
InBlock.gif  Soldier soldier;
ExpandedSubBlockStart.gifContractedSubBlock.gif  
public MagicArmorUpdate(Soldier soldier) dot.gif{
InBlock.gif    
this.soldier = soldier;
ExpandedSubBlockEnd.gif  }

ExpandedSubBlockStart.gifContractedSubBlock.gif  
public String getInfo() dot.gif{
InBlock.gif    
return soldier.getInfo() + "Shanqiu magic update!";
ExpandedSubBlockEnd.gif  }

InBlock.gif
ExpandedSubBlockStart.gifContractedSubBlock.gif  
public int armor() dot.gif{
InBlock.gif    
return soldier.armor() + 5;
ExpandedSubBlockEnd.gif  }

ExpandedBlockEnd.gif}

None.gif
ExpandedBlockStart.gifContractedBlock.gif
public   class  Shanqiu extend Soldier  dot.gif {
ExpandedSubBlockStart.gifContractedSubBlock.gif  
public int armor() dot.gif{
InBlock.gif    
// 初始装甲厚度
InBlock.gif
    return 3;
ExpandedSubBlockEnd.gif  }

ExpandedBlockEnd.gif}

None.gif
ExpandedBlockStart.gifContractedBlock.gif
public   class  PlayingGame  dot.gif {
ExpandedSubBlockStart.gifContractedSubBlock.gif  
public static void main(String args[]) dot.gif{
InBlock.gif    Soldier soldier 
= new Shanqiu();
InBlock.gif    System.out.println(
"Initialized armor:" + soldier.armor());
InBlock.gif
InBlock.gif    
// 普通装甲升级
InBlock.gif
    soldier = new NormalArmorUpdate(soldier);
InBlock.gif    System.out.println(
"Initialized armor:" + soldier.armor());
InBlock.gif
InBlock.gif    
//  使用终极魔法后装甲升级
InBlock.gif
    soldier = new MagicArmorUpdate(soldier);
InBlock.gif    System.out.println(
"Initialized armor:" + soldier.armor());
ExpandedSubBlockEnd.gif  }

ExpandedBlockEnd.gif}


    这样就避免了建立多个Shanqiu类来描述他的不同状态,终极魔法时间过去后,还可以增加一个Decorator来减去相应的装甲厚度,给编程带来了方便。

    Decorator Pattern设计的目的,是为了能够在运行时而不是编译期来动态改变对象的状态,使用组合的方式来增减Decorator,而不是去修改原有的代码来满足业务的需要,这样利于程序的扩展。当然,在Java中使用接口同样能够轻便的实现文中描述的业务需要,但哪种更好,哪种更优?

    Java中的IO提供了大量Decorator Pattern设计的实例,在笔者很久的 一篇随笔中对此进行了简单介绍,建议还是看看JDK的帮助文档。

    参考资料:《Head First Design Pattern》
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值