结构型模式--装饰器模式【Decorator Pattern】

版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/thebigdipperbdx/article/details/79973070

前言

装饰器模式拥有一个设计非常巧妙的结构,它可以动态添加对象功能。
在基本的设计原则中,有一条重要的原则叫做合成/聚合复用原则。
代码复用应尽量使用委托,而不是继承。继承是一种紧密耦合,而委托是松散耦合。
装饰者模式采用的是委托机制。复用组件。

概念

装饰器模式 允许向一个现有的对象添加新的功能,同时又不改变其结构。装饰者可以在所委托被装饰者的行为之前或之后加上自己的行为,以达到特定的目的。

装饰器模式的基本结构如图2.7所示。

这里写图片描述
图2.7 装饰者模式结构

角色

表2.4 装饰者模式角色

角色 作用
组件接口 组件接口是装饰者和被装饰者的超类或者接口。它定义了被装饰者的核心功能和装饰者需要加强的功能点
具体组件 具体组件实现了组件接口的核心方法,完成某一个具体的业务逻辑。它也是被装饰的对象
装饰者 实现组件接口,并持有一个具体的被装饰者对象
具体装饰者 具体实现装饰的业务逻辑,即实现了被分离的各个增强功能点。各个具体装饰者是可以相互叠加的,从而可以构成一个功能更强大的组件对象

典型案例

案例1:对输出结果进行增强

比如现在需要将某一结果通过HTML进行发布,那么需要转化成HTML文本,通过HTTP流传,增加HTTP头,安置TCP头等。
装饰器模式,将核心内容构建、HTML文本构造和HTTP头生成分成三个几乎完全独立的组件,并在使用时灵活的进行装配。

这里写图片描述
图2.8 装饰器模式示例

具体代码实现:

组件接口IPacketCreator:

public interface IPacketCreator {
    public String handleContent();
}

具体组件PacketBodyCreator

public class PacketBodyCreator implements IPacketCreator{

    @Override
    public String handleContent() {
        return "Content of Packet";
    }

}

装饰者PacketDecorator

public abstract class PacketDecorator implements IPacketCreator{
    IPacketCreator component;

    public PacketDecorator(IPacketCreator c){
        component=c;
    }
}

具体装饰者PacketHTMLHeaderCreator

public class PacketHTMLHeaderCreator extends PacketDecorator{
    public PacketHTMLHeaderCreator(IPacketCreator c) {
        super(c);
    }

    public String handleContent() {
        StringBuffer sb=new StringBuffer();
        sb.append("<html>");
        sb.append("<body>");
        sb.append(component.handleContent());
        sb.append("</body>");
        sb.append("</html>\n");
        return sb.toString();
    }
}

具体装饰者PacketHTTPHeaderCreator

public class PacketHTTPHeaderCreator extends PacketDecorator{
    public PacketHTTPHeaderCreator(IPacketCreator c) {
        super(c);
    }

    public String handleContent() {
        StringBuffer sb=new StringBuffer();
        sb.append("Cache-Control:no-cache\n");
        sb.append("Date:"+new Date()+"\n");
        sb.append(component.handleContent());
        return sb.toString();
    }
}

测试类Main

public class Main {
    public static void main(String[] args) {
        IPacketCreator pc = new PacketHTTPHeaderCreator(
                new PacketHTMLHeaderCreator(
                        new PacketBodyCreator()));
        System.out.println(pc.handleContent());
    }

}

运行结果:

Cache-Control:no-cache
Date:Wed Apr 18 10:13:26 CST 2018
<html><body>Content of Packet</body></html>

案例2:JDK实现OutputStream和InputStream类族

java.io.InputStream
java.io.OutputStream
java.io.FilterInputStream【volatile】
java.io.FilterOutputStream
java.io.BufferedInputStream【volatile+synchronized】
java.io.BufferedOutputStream【synchronized】

适用场景:

  • 扩展一个类的功能。
  • 动态增加功能,动态撤销。

优缺点:

优点:

  • 装饰类和被装饰类可以独立发展,不会相互耦合
  • 动态的将责任附加到对象身上。

缺点:

  • 多层装饰比较复杂。

参考资料:
[1] 葛一鸣.Java程序性能优化[M].北京:清华大学出版社,2012: 27-31.

阅读更多
想对作者说点什么?

博主推荐

换一批

没有更多推荐了,返回首页