目录:
定义
角色
特征
应用场景
例子
定义:装饰器模式(Decorator Pattern)允许向一个现有的对象添加新的功能,同时又不改变其结构。这种类型的设计模式属于结构型模式,它是作为现有的类的一个包装。
这种模式创建了一个装饰类,用来包装原有的类,并在保持类方法签名完整性的前提下,提供了额外的功能。
角色:
Component:一般是一个抽象类(也有可能不是),是一组有着某种用途类的基类,包含着这些类最基本的特性。
ConcreteComponent:继承自Component,一般是一个有实际用途的类,这个类就是我们以后要装饰的对象。
Decorator:继承自Component,装饰者需要共同实现的接口(也可以是抽象类),用来保证装饰者和被装饰者有共同的超类,并保证每一个装饰者都有一些必须具有的性质,如每一个装饰者都有一个实例变量(instance variable)用来保存某个Component类型的类的引用。
ConcreteDecorator:继承自Decorator,用来装饰Component类型的类(不能装饰抽象类),为其添加新的特性,可以在委托被装饰者的行为完成之前或之后的任意时候。
特征
1,装饰者(decorator)和被装饰(扩展)的对象有着相同的超类(supertype)。
2,我们可以用多个装饰者去装饰一个对象。
3,我们可以用装饰过的对象替换代码中的原对象,而不会出问题(因为他们有相同的超类)。
4,装饰者可以在委托(delegate,即调用被装饰的类的成员完成一些工作)被装饰者的行为完成之前或之后加上他自己的行为。
5,一个对象能在任何时候被装饰,甚至是运行时。
应用场景
1、扩展一个类的功能。
2、动态增加功能,动态撤销。
例子
我们用《Head First 设计模式》里举的星巴克订单的例子来说明。
星巴克提供不同种类的咖啡和咖啡的调料,星巴克需要一些类来描述他们并且能计算出任意一种咖啡和任意几种调料搭配在一起的价格,如果我们用继承为每一种搭配写一个类的话,就会变成下面这个样子。
要在工作中用这种东西,还不如让我们一起狗带(:зゝ∠)。。。。。
当然我们也可把调料都写在作为超类Beverage里,但是这样的话会造成数据的大量冗余,这是一个解决办法,但是还不够好。
如果套用上面所介绍的装饰者模式的结构就是下面这个样子
Beverage(饮料类):相当与Component
HouseBlend、DarkRoast...(混合咖啡类、礁炒咖啡类...):相当于ConcreteComponent
CondimentDecorator(调料装饰者类):相当于Decorator
Milk、Mocha...(牛奶类、摩卡类...):相当于ConcreteDecorator
装饰者模式的特点,一个ConcreteComponent可以被任意个ConcreteDecorator装饰。
结合实例就来解释,一种咖啡可以和任意种调料搭配。
这样我们来应对各种点单的搭配的时候只需要在一种咖啡(ConcreteComponent)加上各种(ConcreteDecorator)就可以完成了,是不是觉得特别方便。
java实现-装饰者模式-星巴克
public abstract class Beverage {
protected String description="";
public String getDescription(){
return description;
}
public abstract double cost();
}
public abstract class CondimentDecorator extends Beverage {
public abstract String getDescription();
}
public class HouseBlend extends Beverage {
public HouseBlend(){
description="House Blend coffee";
}
@Override
public double cost() {
return 4.9;
}
}
public class Milk extends CondimentDecorator {
protected Beverage beverage;
public Milk(Beverage beverage){
this.beverage=beverage;
}
@Override
public String getDescription() {
return beverage.getDescription()+",with milk";
}
@Override
public double cost() {
return 2.3+beverage.cost();
}
}
public class Mocha extends CondimentDecorator {
protected Beverage beverage;
public Mocha(Beverage beverage){
this.beverage=beverage;
}
@Override
public String getDescription() {
return beverage.getDescription()+",with Mocha";
}
@Override
public double cost() {
return 1.2+beverage.cost();
}
}
public class Starbuzz {
public static void main(String[] args) {
Beverage beverage=new Mocha(new Milk(new Mocha(new HouseBlend())));
//如上就是一杯HouseBlend配上两份Mocha和一份Milk
System.out.println(beverage.getDescription()+":"+beverage.cost());
}
}