7种结构型模式之:装饰器模式(Decorator)

1. 什么是装饰器模式

装饰模式能够实现动态的为对象添加功能,是从一个对象外部来给对象添加功能。通常给对象添加功能,要么直接修改对象添加相应的功能,要么派生对应的子类来扩展,抑或是使用对象组合的方式。显然,直接修改对应的类这种方式并不可取。在面向对象的设计中,而我们也应该尽量使用对象组合,而不是对象继承来扩展和复用功能。装饰器模式就是基于对象组合的方式,可以很灵活的给对象添加所需要的功能。装饰器模式的本质就是动态组合。动态是手段,组合才是目的。总之,装饰模式是通过把复杂的功能简单化,分散化,然后再运行期间,根据需要来动态组合的这样一个模式。

注意上文说的两点,简单化,动态组合。


适用装饰者模式场合:

1.当我们需要为某个现有的对象,动态的增加一个新的功能或职责时,可以考虑使用装饰模式。

2.当某个对象的职责经常发生变化或者经常需要动态的增加职责,避免为了适应这样的变化,而增加继承子类扩展的方式,因为这种方式会造成子类膨胀的速度过快,难以控制。



简单来说,就是在添加功能的情况下,又不失灵活,比如上面说到的生成Excel模板,如果以后想把链接页放到内容页前面,那么只需要调整一下组合的顺序,就可以实现了,不用把它的实现代码大段地拷贝过去。

 

2.装饰器的结构

抽象构件(component)角色 :这个角色用来规范被装饰的对象,一般用接口方式给出。

具体构件(concrete  component )角色 :被装饰的类。

装饰(decorator)角色 :持有一个构件对象的实例。并定义一个跟抽象构件一致的接口。

具体 (concrete    decorator   ) 装饰角色 :负责给具体构件添加附加职责的类。在实际使用中多数情况下装饰角色和具体装饰角色可能由一个类来承担。

 

这个结构中最关键的是,装饰角色持有一个构件对象的实例。这样,需要装饰的实例,才能够传入到装饰器中,让装饰器对其进行装饰。同时,在多个装饰器共同装饰的情况下,还可以把前面的装饰器传入到后面的装饰器中,由最后的装饰器调用动作。因为它们实现了同样的接口,这样做是允许的。

例如:对象A,需要装饰器A,B进行装饰。那么,可以把A传给装饰器A,装饰后,再把A传给装饰器B,继续装饰。也可以把A传给装饰器A之后,再把装饰器A传给装饰器B,由B完成所有的装饰动作(实际上只是调用了A的装饰动作,具体实现仍是在装饰器A当中)。

有同学可能会有疑问了,使用装饰器模式,要求被装饰的类型必须和装饰器的类型,实现相同的接口,具有相同的公有方法。对于已经定义好的类型,怎么能做到装饰呢?

比如上面说的场景,对Excel文件进行装饰,一般我们使用POI的开源包,Excel文件对应HSSFWorkBook类型,那么,是不是我们也要去实现HSSFWorkBook实现的接口WorkBook?那WorkBook接口里面没有我们想要的装饰方法声明怎么办?不就用不了了?

实际上这种场景,仍然可以使用装饰器模式,方法就是将WorkBook类型封装到具体构件的角色里,并提供get方法。这样装饰器得到具体构件后,就可以通过get方法获取到真正需要装饰的对象了(在下面的例子中,我将使用StringBuilder类型,原理是一样的)

3. 一个小例子

嗯,是不是看懵逼了?没关系,我们用上面的生成模板的场景,实践一下。

首先,定义一个抽象构件

复制代码
package com.khlin.test;

/**
 * 模板文件类型。包装了文件的内容,{@link #fillContent()} 用于填充内容
 * @author Kingsley
 *
 */
public interface TemplateFile {
    
    StringBuilder getContent();
    
    void fillContent();
}
复制代码

定义具体构件,即被装饰的类

复制代码
package com.khlin.test;

public class ImportTemplateFile implements TemplateFile {

    StringBuilder content = new StringBuilder();
    
    public ImportTemplateFile() {
        content.append("Title: this is an import template.");
    }
    
    @Override
    public StringBuilder getContent() {
        return this.content;
    }

    @Override
    public void fillContent() {
        System.out.println("ImportTemplateFile: i will do nothing.");
    }

}
复制代码

 

第三步,定义一个装饰器角色。通常是一个抽象类,同时持有第一步抽象构件的一个实例,这是关键。

复制代码
package com.khlin.test;

public abstract class FileDecorator implements TemplateFile {

    TemplateFile templateFile;

    public FileDecorator(TemplateFile templateFile) {
        this.templateFile = templateFile;
    }
}
复制代码

最后一步,实现两个具体的装饰器

复制代码
package com.khlin.test;

public class FileAutherDecorator extends FileDecorator {

    public FileAutherDecorator(TemplateFile templateFile) {
        super(templateFile);
    }

    @Override
    public StringBuilder getContent() {
        return this.templateFile.getContent();
    }

    @Override
    public void fillContent() {
        //先用上一个装饰器处理,再用自己的逻辑处理
        this.templateFile.fillContent();
        
        StringBuilder content = this.templateFile.getContent();
        content.append("\r\nAuther: kingsley");
    }

}
复制代码
复制代码
package com.khlin.test;

import java.util.Date;

public class FileDateDecorator extends FileDecorator{

    public FileDateDecorator(TemplateFile templateFile) {
        super(templateFile);
    }

    @Override
    public StringBuilder getContent() {
        return templateFile.getContent();
    }

    @Override
    public void fillContent() {
        //先用上一个装饰器处理,再用自己的逻辑处理
        this.templateFile.fillContent();
        
        StringBuilder content = this.templateFile.getContent();
        content.append("\r\nDate: " + new Date(System.currentTimeMillis()));
    }

}
复制代码

最后,我们来运行一下

 
复制代码
 1 package com.khlin.test;
 2 
 3 public class App {
 4     public static void main(String[] args) {
 5         TemplateFile file = new ImportTemplateFile();
 6 
 7         // 把要装饰的对象传给装饰器
 8         TemplateFile dateDecorator = new FileDateDecorator(file);
 9 
10         /**
11          * 这里可以有两种做法。
12          * 第一种是先调用dateDecorator.fillContent(),先进行装饰,然后再把file传给autherDecorator
13          * ,由它继续装饰。 第二种是把dateDecorator作为参数传给autherDecorator,由后者一次性地全部装饰。
14          * 这里采用第二种。选用这种,要保证在装饰器的装饰方法里面,显式地调用传入参数的装饰方法。
15          */
16         TemplateFile autherDecorator = new FileAutherDecorator(dateDecorator);
17 
18         autherDecorator.fillContent();
19 
20         System.out.println(autherDecorator.getContent());
21     }
22 }
复制代码

 

运行结果:

 


-----------------------------------------------------------再来一个例子

实例:

  • 汉堡基类
    [java]  view plain  copy
    1. package decorator;  
    2.   
    3. public abstract class Humburger {  
    4.       
    5.     protected  String name ;  
    6.       
    7.     public String getName(){  
    8.         return name;  
    9.     }  
    10.       
    11.     public abstract double getPrice();  
    12.   
    13. }  


  • 鸡腿堡类
    [java]  view plain  copy
    1. package decorator;  
    2.   
    3. public class ChickenBurger extends Humburger {  
    4.       
    5.     public ChickenBurger(){  
    6.         name = "鸡腿堡";  
    7.     }  
    8.   
    9.     @Override  
    10.     public double getPrice() {  
    11.         return 10;  
    12.     }  
    13.   
    14. }  


  • 配料的基类
    [java]  view plain  copy
    1. package decorator;  
    2.   
    3. public abstract class Condiment extends Humburger {  
    4.       
    5.     public abstract String getName();  
    6.   
    7. }  


  • 生菜
    [java]  view plain  copy
    1. package decorator;  
    2.   
    3. public class Lettuce extends Condiment {  
    4.       
    5.     Humburger humburger;  
    6.       
    7.     public Lettuce(Humburger humburger){  
    8.         this.humburger = humburger;  
    9.     }  
    10.   
    11.     @Override  
    12.     public String getName() {  
    13.         return humburger.getName()+" 加生菜";  
    14.     }  
    15.   
    16.     @Override  
    17.     public double getPrice() {  
    18.         return humburger.getPrice()+1.5;  
    19.     }  
    20.   
    21. }  


  • 辣椒
    [java]  view plain  copy
    1. package decorator;  
    2.   
    3. public class Chilli extends Condiment {  
    4.       
    5.     Humburger humburger;  
    6.       
    7.     public Chilli(Humburger humburger){  
    8.         this.humburger = humburger;  
    9.           
    10.     }  
    11.   
    12.     @Override  
    13.     public String getName() {  
    14.         return humburger.getName()+" 加辣椒";  
    15.     }  
    16.   
    17.     @Override  
    18.     public double getPrice() {  
    19.         return humburger.getPrice();  //辣椒是免费的哦  
    20.     }  
    21.   
    22. }  


  • 测试
    [java]  view plain  copy
    1. package decorator;  
    2.   
    3. public class Test {  
    4.   
    5.     /** 
    6.      * @param args 
    7.      */  
    8.     public static void main(String[] args) {  
    9.         Humburger humburger = new ChickenBurger();  
    10.         System.out.println(humburger.getName()+"  价钱:"+humburger.getPrice());  
    11.         Lettuce lettuce = new Lettuce(humburger);  
    12.         System.out.println(lettuce.getName()+"  价钱:"+lettuce.getPrice());  
    13.         Chilli chilli = new Chilli(humburger);  
    14.         System.out.println(chilli.getName()+"  价钱:"+chilli.getPrice());  
    15.         Chilli chilli2 = new Chilli(lettuce);  
    16.         System.out.println(chilli2.getName()+"  价钱:"+chilli2.getPrice());  
    17.     }  
    18.   
    19. }  


  • 输出
    [java]  view plain  copy
    1. 鸡腿堡  价钱:10.0  
    2. 鸡腿堡 加生菜  价钱:11.5  
    3. 鸡腿堡 加辣椒  价钱:10.0  
    4. 鸡腿堡 加生菜 加辣椒  价钱:11.5  

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值