装饰器模式
职责
-
动态的为一个对象增加新的功能
-
装饰模式是一种用于代替继承的技术。无需通过继承增加子类就能扩展对象的新功能。使用对象的关联关系代替继承关系,更加灵活,同时避免类型体系的快速膨胀。
实现细节
-
Component抽象构建组件
真实对象和装饰对象有相同的接口。这样,客户端对象就能够以与真实对象相同的方式同装饰对象交互。
-
ConcreteComponent 具体构建组件(真实对象)
io 流中的 FileInputStream、FileOutputStream
-
Decorator 装饰角色
持有一个抽象构件的引用。装饰对象接受所有客户端的请求,并把这些请求转发给真实的对象。
这样就能在真实对象调用前后增加新的功能 -
ConcreteDecorator 具体装饰角色
负责给构件对象添加新的责任
总结
-
装饰器模式也叫包装器模式
-
装饰器模式降低系统的耦合度,可以动态的增加和删除对象的职责,并使得需要装饰的具体构建类和具体装饰类可以独立变化,以便增加新的具体构建类和具体装饰类
优点
-
扩展对象功能,比继承灵活,不会导致类个数的急剧增加
-
可以对一个对象进行多次装饰,创造出不同行为的组合,得到功能更加强大的对象
-
具体构建类和具体装饰类可以独立变化,用户可以根据需要自己增加新的具体构建子类和具体装饰子类
缺点
-
产生大量的小对象。大量对象占用内存,影响性能
-
易出错,调试麻烦
代码示例:
/**
* @desc 组件接口
*/
public interface IPackageCreator {
public String handleContent();
}
被装饰者:
/**
* @desc 被装饰者
*/
public class PackageBodyCreator implements IPackageCreator {
@Override
public String handleContent() {
return "Hello World!";
}
}
抽象类装饰者(将被装饰者引用进来):
/**
* @desc 装饰者核心组件: 告诉子类自己只是其增强作用,核心业务由 component 实现
*/
public abstract class PackageDecorator implements IPackageCreator {
IPackageCreator component;
public PackageDecorator(IPackageCreator component) {
this.component = component;
}
}
具体的装饰者:
/**
* @desc 具体的装饰器,给content 添加 html 标签
*/
public class PacketHTMLHeaderCreator extends PackageDecorator {
public PacketHTMLHeaderCreator(IPackageCreator component) {
super(component);
}
@Override
public String handleContent() {
StringBuilder sb = new StringBuilder();
sb.append("<html>");
sb.append("<body>");
sb.append(component.handleContent());
sb.append("</body>");
sb.append("</html>\n");
return sb.toString();
}
}
/**
* @desc 具体的装饰器,给content 添加 Http Header
*/
public class PacketHTTPHeaderCreator extends PackageDecorator {
public PacketHTTPHeaderCreator(IPackageCreator component) {
super(component);
}
@Override
public String handleContent() {
StringBuilder sb = new StringBuilder();
sb.append("Cache-Control: no-cache\n");
sb.append(component.handleContent());
return sb.toString();
}
}
客户端调用:
public class Main {
public static void main(String[] args) {
IPackageCreator creator = new PacketHTTPHeaderCreator(new PacketHTMLHeaderCreator(new PackageBodyCreator()));
System.out.println(creator.handleContent());
}
}
-----------------------------------------------
Cache-Control: no-cache
<html><body>Hello World!</body></html>