业务场景
当我们需要将一段文字发布到网页上进行显示时,我们需要对网页进行HMTL格式化,然后加上HTTP头部等一系列的操作,如果突然有一天,我们需要修改HMTL格式时候,就……
OK,其实我们完全可以使用装饰模式,将上面繁琐的方式一步步分解,方便后期维护
- 首先我们使用PacketBodyCreator来收集网页内容
- 其次我们使用PacketHTMLHeaderCreator来讲内容格式化为HTML
- 然后我们使用PacketHTTPHeaderCreator来将HTML文本加上HTTP头部
- 我们为上面三个接口定义了一个共同的接口IPacketCreator
- 另外还有一个抽象的父类PacketDecorator,他的作用是
PacketHTTPHeaderCreator可以增强PacketHTMLHeaderCreator,
同理PacketHTMLHeaderCreator增强PacketBodyCreator
代码
- 先看接口
public interface IPacketCreator {
public String handleContent();
}
- 抽象的父类PacketDecorator:实现它就能增强别的类,前提是这个类要实现IPacketCreator接口
public abstract class PacketDecorator implements IPacketCreator{
IPacketCreator componet;
public PacketDecorator(IPacketCreator c){
componet=c;
}
}
- PacketBodyCreator:收集内容,实现了IPacketCreator 接口,可以被增强
public class PacketBodyCreator implements IPacketCreator{
@Override
public String handleContent() {
return "Content of Packet";
}
}
- PacketHTMLHeaderCreator:格式化为HTML,继承了PacketDecorator父类,也相当于实现了IPacketCreator 接口
public class PacketHTMLHeaderCreator extends PacketDecorator{
public PacketHTMLHeaderCreator(IPacketCreator c) {
super(c);
}
@Override
public String handleContent() {
StringBuffer sb=new StringBuffer();
sb.append("<html>");
sb.append("<body>");
sb.append(componet.handleContent());
sb.append("</body>");
sb.append("</html>\n");
return sb.toString();
}
}
- PacketHTTPHeaderCreator:方式与PacketHTMLHeaderCreator差不多,为别的类添加HTTP头部
public class PacketHTTPHeaderCreator extends PacketDecorator{
public PacketHTTPHeaderCreator(IPacketCreator c) {
super(c);
}
@Override
public String handleContent() {
StringBuffer sb=new StringBuffer();
sb.append("Cache-Control:no-cache\n");
sb.append("Date:Mon,31Dec201204:25:57GMT\n");
sb.append(componet.handleContent());
return sb.toString();
}
}
- 那怎么调用呢?这下面Main方法
public static void main(String[] args) {
IPacketCreator pc=new PacketHTTPHeaderCreator(
new PacketHTMLHeaderCreator(
new PacketBodyCreator()));
System.out.println(pc.handleContent());
}
总结
通过上面,我们可以得出下面的结论
- 装饰着模式可以动态添加对象功能,比如使用普通的文本加上HTML的标签
- 他符合:合成/聚合复用原则,最后会讲什么是合成/聚合复用原则
- 装饰者模式可以有效分离各个组件,从而提升模块的可维护性并增加模块的复用性
提高性能?
听到这里,你是不是疑惑,不是说装饰模式可以优化程序性能吗? 那我们来看一下JDK的实现,看他如何优化程序性能的
JDK利用装饰模式优化OutPutStream的性能:
- OutPutStream:功能简单,且比较弱
- 我们可以利用FileOutPutStream,为它实现文件读取功能
- 我们再利用DataOutPutStream增加多种数据类型写的操作
- 最后,我们可以利用BufferedOutPutStream增加缓存功能,从而增强I/O性能
BufferedOutPutStream作用,当数据比较少时,就写入缓存,当数据积累到一定时候,才一次性将缓存内容写入文档中
所以,装饰模式也可以提高程序性能
补充
什么是合成/聚合复用原则?
代码复用应该尽可能使用委托,而不是继承,因为继承是一种紧密耦合,任何父类的改动都会影响其子类,不利于系统维护。而委托则是松耦合,只要接口不变,委托类的改动都不会影响其上层对象