设计模式之---装饰者模式(Decorator Design Pattern)

意图
装饰模式是在不必改变原类文件和使用继承的情况下,动态地扩展一个对象的功能。
它是通过创建一个包装对象,也就是装饰来包裹真实的对象。
特点
(1) 装饰对象和真实对象有相同的接口。这样客户端对象就能以和真实对象相同的方式和装饰对象交互。
(2) 装饰对象包含一个真实对象的引用(reference)
(3) 装饰对象接受所有来自客户端的请求。它把这些请求转发给真实的对象。
(4) 装饰对象可以在转发这些请求以前或以后增加一些附加功能。这样就确保了在运行时,不用修改给定对象的结构就可以在外部增加附加的功能。在面向对象的设计中,通常是通过继承来实现对给定类的功能扩展。
使用场景
1. 需要扩展一个类的功能,或给一个类添加附加职责。
2. 需要动态的给一个对象添加功能,这些功能可以再动态的撤销。
3. 需要增加由一些基本功能的排列组合而产生的非常大量的功能,从而使继承关系变的不现实。
4. 当不能采用生成子类的方法进行扩充时。一种情况是,可能有大量独立的扩展,为支持每一种组合将产生大量的子类,使得子类数目呈爆炸性增长。另一种情况可能是因为类定义被隐藏,或类定义不能用于生成子类。

设计原则
1. 多用组合,少用继承。
利用继承设计子类的行为,是在编译时静态决定的,而且所有的子类都会继承到相同的行为。然而,如果能够利用组合的做法扩展对象的行为,就可以在运行时动态地进行扩展。
2. 类应设计的对扩展开放,对修改关闭。

使用实例
角色
  (1)抽象构件(Component)角色:给出一个抽象接口,以规范准备接收附加责任的对象。
  (2)具体构件(Concrete Component)角色:定义一个将要接收附加责任的类。
  (3)装饰(Decorator)角色:持有一个构件(Component)对象的实例,并实现一个与抽象构件接口一致的接口。
  (4)具体装饰(Concrete Decorator)角色:负责给构件对象添加上附加的责任。
以下示例中,
ThirdParty.Java假定是一个现有的或者第三方的功能,因某种原因我们不能直接修改,它提供了一个sayMsg()的方法,
而我们现在要做的是想在它的sayMsg()方法中增加一些我们想额外输出的内容,于是我们重写了一个Decorator.java类。
MailTest.java是客户端测试程序。

/* * 抽象接口类 */
public interface IthirdParty {
    public String sayMsg();
}
/* * 具体类 */
public class ThirdParty implements IthirdParty {
    public String sayMsg() {
        return "hello";
  }
}
/* * 具体装饰类1 */
public class Decorator1 implements IThirdParty {
    private IThirdParty thirdParty;

    public Decorator1(IThirdParty thirdParty) {
        this.thirdParty = thirdParty;
    }

    public String sayMsg() {
        return "##1" + thirdParty.sayMsg() + "##1";
    }
}
/* * 具体装饰类2 */
public class Decorator2 implements IThirdParty {
    private IThirdParty thirdParty;

    public Decorator2(IThirdParty thirdParty) {
        this.thirdParty = thirdParty;
    }

    public String sayMsg() {
        return "##2" + thirdParty.sayMsg() + "##2";
    }
}
/* * 测试类 */
public class MailTest {
    public static void main(String[] args){

      IthirdParty thirdPartyOne =new ThirdParty();
      IthirdParty decorator1 =new Decorator1(thirdPartyOne);
      IthirdParty decorator2 =new Decorator2(decorator1);
      
      System.out.println(decorator2.sayMsg());
    }
}

执行结果是:

##2##1hello##1##2

你可能感兴趣的:(设计模式)