Decorator Pattern
1. 什么是装饰模式
装饰模式(Decorator Pattern):动态地给一个对象增加一些额外的职责,就增加对象功能来说,装饰模式比生成子类实现更为灵活。
装饰模式是一种对象结构型模式。
装饰类的引入将大大简化本系统的设计,它也是装饰模式的核心。
装饰模式可以在不改变一个对象本身功能的基础上给对象增加额外的新行为。
装饰模式是一种用于替代继承的技术,它通过一种无须定义子类的方式来给对象动态增加职责,使用对象之间的关联关系取代类之间的继承关系。
在装饰模式中引入了装饰类,在装饰类中既可以调用待装饰的原有类的方法,还可以增加新的方法,以扩充原有类的功能。
2. 装饰模式类角色解析
装饰模式结构图:
- Component(抽象构件):它是具体构件和抽象装饰类的共同父类,声明了在具体构件中实现的业务方法,它的引入可以使客户端以一致的方式处理未被装饰的对象以及装饰之后的对象,实现客户端的透明操作。
- ConcreteComponent(具体构件):它是抽象构件类的子类,用于定义具体的构件对象,实现了在抽象构件中声明的方法,装饰器可以给它增加额外的职责(方法)。
- Decorator(抽象装饰类):它也是抽象构件类的子类,用于给具体构件增加职责,但是具体职责在其子类中实现。它维护一个指向抽象构件对象的引用,通过该引用可以调用装饰之前构件对象的方法,并通过其子类扩展该方法,以达到装饰的目的。
- ConcreteDecorator(具体装饰类):它是抽象装饰类的子类,负责向构件添加新的职责。每一个具体装饰类都定义了一些新的行为,它可以调用在抽象装饰类中定义的方法,并可以增加新的方法用以扩充对象的行为。
3. 装饰模式好处
- 对于扩展一个对象的功能,装饰模式比继承更加灵活性,不会导致类的个数急剧增加。
- 可以对一个对象进行多次装饰,通过使用不同的具体装饰类以及这些装饰类的排列组合,可以创造出很多不同行为的组合,得到功能更为强大的对象。
- 具体构件类与具体装饰类可以独立变化,用户可以根据需要增加新的具体构件类和具体装饰类,原有类库代码无须改变,符合“开闭原则”。
4. 代理模式和装饰模式区别
代理模式和装饰模式非常类似,甚至代码都类似。二者最主要的区别是:代理模式中,代理类对被代理的对象有控制权,决定其执行或者不执行。而装饰模式中,装饰类对代理对象没有控制权,只能为其增加一层装饰,以加强被装饰对象的功能,仅此而已。
代理模式和装饰模式的最大区别就是目的: 代理模式控制的是对象的访问控制,而装饰模式仅仅是装饰一下,增加一些功能。
5. 实操感想
桥接模式是两个独立的抽象体系,而装饰模式是在一个体系内进行功能加强。避免了多继承带来的冗余。
6. 代码示例
有一套图形界面构件库VisualComponent,该构件库提供了大量基本构件,如窗体、文本框、列表框等,由于在使用该构件库时,用户经常要求定制一些特效显示效果,如带滚动条的窗体、带黑色边框的文本框、既带滚动条又带黑色边框的列表框等等,因此经常需要对该构件库进行扩展以增强其功能.
//抽象构件
public abstract class Component {
public abstract void display();
}
//具体构件
public class ComponentWindow extends Component {
@Override public void display() {
System.out.println("显示窗体");
}
@Override public String toString() {
return "Window窗体";
}
}
//具体构件
public class ComponentTextArea extends Component {
@Override public void display() {
System.out.println("显示文本框");
}
}
//抽象装饰类
public abstract class DecoratorComponent extends Component {
protected Component component;
public DecoratorComponent(Component component) {
this.component = component;
}
@Override public void display() {
component.display();
}
}
//具体装饰类,添加边框
public class DecoratorBorder extends DecoratorComponent {
public DecoratorBorder(Component component) {
super(component);
}
@Override public void display() {
System.out.println(component.toString() + " 添加边框");
super.display();
}
}
//具体装饰类,添加滚动条
public class DecoratorScrollBar extends DecoratorComponent {
public DecoratorScrollBar(Component component) {
super(component);
}
@Override public void display() {
System.out.println("添加滚动条");
super.display();
}
}
//测试类,给Window添加边框
public class MainClass {
public static void main(String[] args) {
ComponentWindow componentWindow = new ComponentWindow();
DecoratorBorder decoratorBorder = new DecoratorBorder(componentWindow);
decoratorBorder.display();
}
}