一、定义
装饰模式:动态地给一个被装饰者对象添加其他兄弟类一些额外的职责,但是不改变被装饰者类的功能。就增加功能来说,装饰模式相比生成子类更为灵活。装饰模式中,通过装饰者给被装饰者扩展了功能。
二、使用场景
- 需要扩展一个类的功能,或给一个类增加附加功能。
- 需要动态地给一个对象增加功能,这些功能可以再动态地撤销。
- 需要为一批的兄弟类进行改装或加装功能,当然是首选装饰模式。
- Java 中的 IO 流的架构。
- Android 中的 Context 家族。
三、栗子
- 被装饰者及其抽象
/**
* 抽象的被装饰类
* @author innovator
*
*/
public abstract class Component {
public abstract void operate();
}
/**
* 具体的被装饰者类实现
* @author innovator
*
*/
public class ConcreteComponent extends Component{
@Override
public void operate() {
System.out.println("被装饰者类----> operate");
}
}
- 装饰者及其抽象
/**
* 抽象的装饰者类
* @author innovator
*
*/
public class Decorator extends Component{
//引用被装饰者
private Component component;
public Decorator(Component c) {
this.component = c;
}
@Override
public void operate() {
this.component.operate();
}
}
/**
* 具体的装饰者类实现
* @author innovator
*
*/
public class ConcreteDecorator1 extends Decorator{
public ConcreteDecorator1(Component c) {
super(c);
}
@Override
public void operate() {
decorate1();
super.operate();
}
public void decorate1() {
System.out.println("装饰者1--------> 修饰方法1");
}
}
public class ConcreteDecorator2 extends Decorator{
public ConcreteDecorator2(Component c) {
super(c);
}
@Override
public void operate() {
decorate2();
super.operate();
}
public void decorate2() {
System.out.println("装饰者2--------> 修饰方法2");
}
}
- 场景
public class Client {
public static void main(String[] args) {
Component component = new ConcreteComponent();
//被装饰者1装饰
component = new ConcreteDecorator1(component);
component = new ConcreteDecorator2(component);
Component component2 = new ConcreteDecorator2(new ConcreteDecorator1(new ConcreteComponent()));
component.operate();
component2.operate();
}
}
- 结果
装饰者2--------> 修饰方法2
装饰者1--------> 修饰方法1
被装饰者类----> operate
装饰者2--------> 修饰方法2
装饰者1--------> 修饰方法1
被装饰者类----> operate
四、优点
- 装饰模式可以动态地扩展一个实现类的功能。
- 装饰模式是继承关系的一个替代方案。不管装饰多少层,最终返回的对象还是抽象类。
- 装饰类和被装饰类可以独立发展,而不会相互耦合。
五、缺点
- 多层的装饰会造成大量的小类
- 应该尽量减少装饰类的数量,以便降低系统的复杂度。