说到装饰,我们首先能想到的是房间中的各种陈设和布景。在房间中,每增加一个装饰物品,整个房间的氛围都会有所改变。而我们的装饰者模式也起到了这么一个作用:
Attach additional responsibilities to an object dynamically. Decorators provide a flexible alternative to subclassing for extending functionality.
动态地为一个对象增加其他功能。装饰者模式提供了一种灵活的方案来替代子类的继承。
我们来看一下装饰者模式的框架:
- Component: 即是我们需要装饰的对象,Component是抽象的,定义了该对象的方法。实际上我们要装饰的不是这个对象,而是这个对象的方法,在方法被调用前或后增加其他逻辑,或实现其他效果;
- ConcreteComponent: 即是上述对象子类化后的类,实现了抽象方法;
- Decorator: 装饰者,也继承了Component类,同样需要定义相同的方法;
- ConcreteDecoratorA & B: 具体的装饰者类,覆盖了Component的方法,增加了装饰逻辑;
其实说白了,就是使用合理的覆盖来实现多层次的方法叠加,达到装饰原始内部方法的效果。
我们来看一个例子,使用装饰者模式,将一个普通的文字装饰成斜体(Italic)+粗体(Bold)的字体。
Component
我们首先定义需要进行装饰的抽象类,它是一个文本,定义了获得文本字体的方法getFont()
。
public abstract class MockText {
public abstract Font getFont();
}
ConcreteComponent
然后我们需要一个实现类来继承MockText:
ublic class ConcreteText extends MockText {
private Font font = new Font(Font.SANS_SERIF, Font.PLAIN, 12);
@Override
public Font getFont() {
return font;
}
Decorator
而后我们需要一个抽象的装饰者,这个装饰者需要继承Component以获得同样的行为。它使用一个MockText作为构造函数的参数:
public abstract class FontDecorator extends MockText {
private MockText text;
public FontDecorator(MockText text) {
this.text = text;
}
@Override
public Font getFont() {
return text.getFont();
}
}
ConcreteDecorator
而后我们需要两个具体的Decorator实现类,用来将字体变为斜体和粗体:
public class BoldTextDecorator extends FontDecorator {
public BoldTextDecorator(MockText text) {
super(text);
}
@Override
public Font getFont(){
Font font = super.getFont();
return new Font(font.getName(), font.getStyle() | Font.BOLD,font.getSize());
}
}
public class ItalicTextDecorator extends FontDecorator{
public ItalicTextDecorator(MockText text) {
super(text);
}
@Override
public Font getFont(){
Font font = super.getFont();
return new Font(font.getName(), font.getStyle() | Font.ITALIC,font.getSize());
}
}
最后我们使用一个main函数来驱动这个框架:
public static void main(String[] args){
MockText text= new ConcreteText();
System.out.println(text.getFont());
text = new BoldTextDecorator(text);
text = new ItalicTextDecorator(text);
System.out.println(text.getFont());
}
我们尝试打印了两次text.getFont的结果,结果如下:
java.awt.Font[family=SansSerif,name=SansSerif,style=plain,size=12]
java.awt.Font[family=SansSerif,name=SansSerif,style=bolditalic,size=12]
可以看到第一次的style还是plain,而第二次的style已经变成了bolditalic