Java设计模式之-装饰者模式(Decorator)

说到装饰,我们首先能想到的是房间中的各种陈设和布景。在房间中,每增加一个装饰物品,整个房间的氛围都会有所改变。而我们的装饰者模式也起到了这么一个作用:

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

你可能感兴趣的:(Java设计模式之-装饰者模式(Decorator))