Eason-S

导航

每天一个设计模式(3):装饰者模式

3.装饰者模式

  装饰者模式动态地将责任附加到对象上。若要扩展功能,装饰者提供了比继承更有弹性的替代方案。

一.问题引入

  咖啡店的类设计:

  一个饮料基类,各种饮料类继承这个基类,并且计算各自的价钱。

  饮料中需要加入各种调料,考虑在基类中加入一些布尔值变量代表是否加入各种调料,基类的cost()中的计算各种调料的价钱,子类覆盖cost(),并且在其中调用超类的cost(),加上特定饮料的价钱,计算出子类特定饮料的价钱。

  缺点:类数量爆炸、基类加入的新功能并不适用于所有的子类、调料价钱的改变、新调料的出现都会要求改变现有代码;有的子类并不适合某些调料等情况……

二.要点

  1.类应该对扩展开放,对修改关闭。

  我们的目标是允许类容易扩展,在不修改现有代码的情况下,就可搭配新的行为。好处是:这样的设计具有弹性可以应对改变,可以接受新的功能来应对改变的需求。

  2.装饰者和被装饰对象有相同的超类型。

  2.组合和委托可用于在运动时动态的加上新的行为。

  3.装饰者可以在被装饰者的行为前面/后面加上自己的行为,甚至将被装饰者的行为整个取代掉,而达到特定的目的。

  4.可以用无数个装饰者包装一个组件。

  5.装饰者一般对组件的客户是透明的,除非客户程序依赖于组件的具体类型。

  6.装饰者会导致设计中出现许多小对象,如果过度使用,会让程序变得很复杂。 

三.用装饰者模式解决问题

  解决咖啡店饮料问题的方法:

  以饮料为主体,然后在运行时以调料来“装饰”饮料。

  比如,顾客想要摩卡(Mocha)和奶泡(Whip)深焙咖啡(DarkRoast):

  DarkRoast继承自Beverage,有一个cost()方法。

  第一步,以DarkRoast对象开始;

  第二步,顾客想要摩卡,所以建立一个Mocha装饰者对象,并用它将DarkRoast对象包装(wrap)起来;

  第三步,顾客想要奶泡,所以建立一个Whip装饰者对象,并用它将Mocha对象包起来;(Mocha和Whip也继承自Beverage,有一个cost()方法);

  最后,为顾客算钱,通过调用最外圈装饰者(Whip)的cost()就可以。Whip()的cost()会先委托它装饰的对象(Mocha)计算出价钱,然后在加上奶泡的价钱。Mocha的cost()也是类似。

四.UML关系图

  Source类是被装饰类,Decorator类是一个装饰类,可以为Source类动态的添加一些功能。

五.实现代码

Sourceable接口:

public interface Sourceable {  
    public void method();  
} 

被装饰类:

public class Source implements Sourceable {  
  
    @Override  
    public void method() {  
        System.out.println("the original method!");  
    }  
}  

装饰类:

public class Decorator implements Sourceable {  
  
    private Sourceable source;  
      
    public Decorator(Sourceable source){  
        super();  
        this.source = source;  
    }  
    @Override  
    public void method() {  
        System.out.println("before decorator!");  
        source.method();  
        System.out.println("after decorator!");  
    }  
}  

测试类:

public class DecoratorTest {  
  
    public static void main(String[] args) {  
        Sourceable source = new Source();  
        Sourceable obj = new Decorator(source);  
        obj.method();  
    }  
}  

输出:

before decorator!
the original method!
after decorator!

六.应用场景

1.需要扩展一个类的功能。
2.动态的为一个对象增加功能,而且还能动态撤销。(继承不能做到这一点,继承的功能是静态的,不能动态增删。)
缺点:产生过多相似的对象,不易排错!

七.扩展

《Head First设计模式》中的实现类图:

 

java.io包内的装饰者模式实现类图:

 

 

参考:

《Head First设计模式》

http://blog.csdn.net/zhangerqing/article/details/8239539

http://www.cnblogs.com/mengdd/archive/2013/01/03/2843439.html

posted on 2016-07-11 15:40  Eason_S  阅读(370)  评论(0编辑  收藏  举报