装饰器模式使用分层对象来动态透明地向单个对象添加责任。装饰器指定包装在最初的对象周围的所有对象具有相同的接口。
Decorator装饰器,顾名思义,就是动态地给一个对象添加一些额外的职责,就好比为房子进行装修一样。因此,装饰器模式具有如下的特征:
它必须具有一个装饰的对象。
它必须拥有与被装饰对象相同的接口。
它可以给被装饰对象添加额外的功能。
用一句话总结就是:保持接口,增强性能。
装饰器通过包装一个装饰对象来扩展其功能,而又不改变其接口,这实际上是基于对象的适配器模式的一种变种。它与对象的适配器模式的异同点如下。
下面给出一个装饰器模式的示例:
class Basic {
private String value;
public void set(String val){value = val;}
public String get(){ return value; }
}
一个基类,定义了一个set()和一个get()方法;
class Decorator extends Basic {
protected Basic basic;
public Decorator(Basic basic){ this.basic = basic; }
public void set(String val){ basic.set(val); }
public String get(){ return basic.get(); }
}
类Decorator是基类Basic的一个实现。接下来实现类TimeStamped继承Decorator实现性能增加;
class TimeStamped extends Decorator {
private final long timeStamp;
public TimeStamped(Basic basic) {
super(basic);
timeStamp = new Date().getTime();
}
public long getStamp(){ return timeStamp; }
}
装饰器TimeStamped采用典型的对象适配器模式,他拥有一个TimeStamped构造函数,并以Basic为参数,以期Decorator保持相同的接口。并实现自己特有的方法getStamp().这就是装饰器扩展的功能。
class SerialNumbered extends Decorator {
private static long counter = 1;
private final long serialNumber = counter++ ;
public SerialNumbered(Basic basic) {
super(basic);
}
public long getSerialNumber(){ return serialNumber; }
}
类SerialNumbered是另一个装饰器,不过就是所装饰的功能不一样,这里获取自增长的serialNumber数值。
实现了两个装饰器,这时,我们可以使用对象的适配器模式一样来使用这些装饰器,使用不同的装饰器就可以达到不同的装饰效果。
TimeStamped t = new TimeStamped(new Basic());
TimeStamped t2 = new TimeStamped(new SerialNumbered(new Basic()));
t2.getStamp();
SerialNumbered s = new SerialNumbered(new Basic());
SerialNumbered s2 = new SerialNumbered(new TimeStamped(new Basic()));
s2.getSerialNumber();
分别用TimeStamped,SerialNumbered装饰源类对象Basic,和类对象SerialNumbered。