设计模式(14)——装饰者模式

装饰模式

一、概述

定义: 动态地给一个对象添加一些额外的职责,就增加功能来说,装饰模式比生成子类更灵活。

原理:增加一个修饰类包裹原来的类,包裹的方式一般是通过在将原来的对象作为修饰类的构造函数的参数。装饰类实现新的功能,但是,在不需要用到新功能的地方,它可以直接调用原来的类中的方法。因为修饰类和原来的类有相同的接口。

装饰模式以对客户端透明的方式动态的给一个对象附加上更多的责任。换言之客户端并不会觉的对象在装饰前和装饰后有什么区别。  装饰模式可以在不创造更多的子类的模式下,将对象的功能加以扩展

二、适用性:   

  • 需要扩展一个类的功能,或给一个类增加附加责任。
  • 需要动态地给一个对象增加功能,这些功能可以再动态地撤销。
  • 需要增加由一些基本功能的排列组合而产生的非常大量的功能,从而使继承关系变得不现实。 

三、引入例子  装饰成绩单骗取家长签字

对成绩单进行装饰,骗取家长签字。

个成绩单的类图

成绩单的抽象类

[java]  view plain copy print ?
  1. public abstract class SchoolReport {  
  2.     // 成绩单的主要展示的就是你的成绩情况  
  3.     public abstract void report();  
  4.   
  5.     // 成绩单要家长签字,这个是最要命的  
  6.     public abstract void sign(String name);  
  7. }  

四年级的成绩单实现类

[java]  view plain copy print ?
  1. /** 
  2.  * 四年级的成绩单 
  3.  */  
  4. public class FouthGradeSchoolReport extends SchoolReport {  
  5.     // 我的成绩单  
  6.     public void report() {  
  7.         // 成绩单的格式是这个样子的  
  8.         System.out.println("尊敬的XXX家长:");  
  9.         System.out.println(" ......");  
  10.         System.out.println(" 语文 62 数学65 体育 98 自然 63");  
  11.         System.out.println(" .......");  
  12.         System.out.println(" 家长签名: ");  
  13.     }  
  14.   
  15.     // 家长签名  
  16.     public void sign(String name) {  
  17.         System.out.println("家长签名为:" + name);  
  18.     }  
  19.   
  20. }  

老爸开始看最真实的成绩单,

[java]  view plain copy print ?
  1. /** 
  2.  * 老爸看原始成绩单 
  3.  */  
  4. public class Father {  
  5.     public static void main(String[] args) {  
  6.         // 成绩单拿过来  
  7.         SchoolReport sr = new FouthGradeSchoolReport();  
  8.         // 看成绩单  
  9.         sr.report();  
  10.         // 签名?休想!  
  11.     }  
  12. }  

结果就是老爸不给签名

为了骗取签名,我没有直接把成绩单交给家长,而是在交给他之前做了点技术工作,我要把成绩单封装一下,封装分类两步走:

第一步:跟老爸说各个科目的最高分,语文最高是75,数学是78,自然是80,然老爸觉的我成绩与最高分数相差不多,(实情是基本上都集中在70分以上,我这60多分基本上还是垫底的角色)

 

第二步:在老爸看成绩单后,告诉他我是全班排名第38名,(实情是有将近十个同学退学了!这个情况我是不说的。反正成绩单上没有写总共有多少同学,排名第几名等等)

 

使用继承的方法

我想这是你最容易想到的类图,通过直接增加了一个子类,重写report方法,很容易的解决了这个问题,

[java]  view plain copy print ?
  1. public class SugarFouthGradeSchoolReport extends FouthGradeSchoolReport {  
  2.     // 首先要定义你要美化的方法,先给老爸说学校最高成绩  
  3.     private void reportHighScore() {  
  4.         System.out.println("这次考试语文最高是75,数学是78,自然是80");  
  5.     }  
  6.   
  7.     // 在老爸看完毕成绩单后,我再汇报学校的排名情况  
  8.     private void reportSort() {  
  9.         System.out.println("我是排名第38名...");  
  10.     }  
  11.   
  12.     // 由于汇报的内容已经发生变更,那所以要重写父类  
  13.     @Override  
  14.     public void report() {  
  15.         this.reportHighScore(); // 先说最高成绩  
  16.         super.report(); // 然后老爸看成绩单  
  17.         this.reportSort(); //然后告诉老爸学习学校排名  
  18.     }  
  19. }  

Father类稍做修改就可以看到美化后的成绩单

[java]  view plain copy print ?
  1. public class Father {  
  2.     public static void main(String[] args) {  
  3.         // 美化过的成绩单拿过来  
  4.         SchoolReport sr = new SugarFouthGradeSchoolReport();  
  5.         // 看成绩单  
  6.         sr.report();  
  7.         // 然后老爸,一看,很开心,就签名了  
  8.         sr.sign("老三"); // 获得签名  
  9.     }  
  10. }  

获得签名

存在的问题

通过继承确实能够解决这个问题,老爸看成绩单很开心,然后就给签字了,但是现实的情况很复杂的,可能老爸听我汇报最高成绩后,就直接乐开花了,直接签名了,后面的排名就没必要了,或者老爸要先听排名情况,那怎么办?继续扩展类?你能扩展多少个类?这还是一个比较简单的场景,一旦需要装饰的条件非常的多,比如20个,你还通过继承来解决,你想想的子类有多少个?你是不是马上就要崩溃了!

好,你也看到通过继承情况确实出现了问题,类爆炸,类的数量激增,

装饰模式来解决这些问题

增加一个抽象类和两个实现类,其中Decorator的作用是封装SchoolReport类,Decorator抽象类的目的很简单,就是要让子类来对封装SchoolReport的子类,怎么封装?重写report方法。,先调用具体装饰类的装饰方法reportHighScore,然后再调用具体构件的方法。

Decorator

[java]  view plain copy print ?
  1. /** 
  2.  * 装饰类,我要把我的成绩单装饰一下 
  3.  */  
  4. public abstract class Decorator extends SchoolReport {  
  5.     // 首先我要知道是那个成绩单  
  6.     private SchoolReport sr;  
  7.   
  8.     // 构造函数,传递成绩单过来  
  9.     public Decorator(SchoolReport sr) {  
  10.   
  11.         this.sr = sr;  
  12.     }  
  13.   
  14.     // 成绩单还是要被看到的  
  15.     public void report() {  
  16.         this.sr.report();  
  17.     }  
  18.   
  19.     // 看完毕还是要签名的  
  20.     public void sign(String name) {  
  21.         this.sr.sign(name);  
  22.     }  
  23. }  

HighScoreDecorator实现类:

[java]  view plain copy print ?
  1. /** 
  2.  *  我要把我学校的最高成绩告诉老爸 
  3.  */  
  4. public class HighScoreDecorator extends Decorator {  
  5.     // 构造函数  
  6.     public HighScoreDecorator(SchoolReport sr) {  
  7.         super(sr);  
  8.     }  
  9.   
  10.     // 我要汇报最高成绩  
  11.     private void reportHighScore() {  
  12.         System.out.println("这次考试语文最高是75,数学是78,自然是80");  
  13.     }  
  14.   
  15.     // 最高成绩我要做老爸看成绩单前告诉他,否则等他一看,就抡起笤帚有揍我,我那还有机会说呀  
  16.     @Override  
  17.     public void report() {  
  18.         this.reportHighScore();  
  19.         super.report();  
  20.   
  21.     }  
  22. }  

SortDecorator实现类:

[java]  view plain copy print ?
  1. /** 
  2.  * 排名的情况汇报 
  3.  */  
  4. public class SortDecorator extends Decorator {  
  5.     // 构造函数  
  6.     public SortDecorator(SchoolReport sr) {  
  7.         super(sr);  
  8.     }  
  9.   
  10.     // 告诉老爸学校的排名情况  
  11.     private void reportSort() {  
  12.         System.out.println("我是排名第38名...");  
  13.     }  
  14.   
  15.     // 老爸看完成绩单后再告诉他,加强作用  
  16.     @Override  
  17.     public void report() {  
  18.         super.report();  
  19.         this.reportSort();  
  20.     }  
  21. }  

老爸一看成绩单

[java]  view plain copy print ?
  1. public class Father {  
  2.     public static void main(String[] args) {  
  3.         // 成绩单拿过来  
  4.         SchoolReport sr;  
  5.         sr = new FouthGradeSchoolReport(); // 原装的成绩单  
  6.         // 加了最高分说明的成绩单  
  7.         sr = new HighScoreDecorator(sr);  
  8.         // 又加了成绩排名的说明  
  9.         sr = new SortDecorator(sr);  
  10.         // 看成绩单  
  11.         sr.report();  
  12.         // 然后老爸,一看,很开心,就签名了  
  13.         sr.sign("老三"); //获得签名  
  14.     }  
  15. }  

四、模式解读

4.1、类图

4.2、角色

1) 抽象构建角色(Component):给出一个抽象的接口,以规范准备接受附加责任的对象。

2) 具体的构建角色(ConcreteComponent):定义一个将要接受附加责任的类。

3) 装饰角色(Docorator):持有一个抽象构建(Component)角色的引用,并定义一个与抽象构件一致的接口。

4) 具体的装饰角色(ConcreteDecorator):负责给构建对象“贴上”附加的责任。

4.3、一般代码

1)   抽象的构建接口:

[java]  view plain copy print ?
  1. public interface Component {  
  2.     public void doSomething();  
  3. }  

2)   具体的构建角色:

[java]  view plain copy print ?
  1. public class ConcreteComponent implements Component {  
  2.     @Override  
  3.     public void doSomething() {  
  4.         System.out.println("功能A");  
  5.     }  
  6. }  

3)   装饰角色:

[java]  view plain copy print ?
  1. public class Decorate implements Component {  
  2.     private Component component;  
  3.   
  4.     public Decorate(Component component) {  
  5.         this.component = component;  
  6.     }  
  7.   
  8.     @Override  
  9.     public void doSomething() {  
  10.         component.doSomething();  
  11.     }  
  12. }  

4)   具体装饰角色1

[html]  view plain copy print ?
  1. public class ConcreteDecorate1 extends Decorate {  
  2.     public ConcreteDecorate1(Component component) {  
  3.         super(component);  
  4.     }  
  5.   
  6.     @Override  
  7.     public void doSomething() {  
  8.         super.doSomething();  
  9.   
  10.         this.doAnotherDosomething();  
  11.     }  
  12.   
  13.     private void doAnotherDosomething() {  
  14.         System.out.println("功能B");  
  15.     }  
  16. }  

5)   具体装饰角色2

[java]  view plain copy print ?
  1. public class ConcreteDecorate2 extends Decorate  
  2. {  
  3.    public ConcreteDecorate2(Component component)  
  4.    {  
  5.       super(component);  
  6.    }  
  7.     
  8.    @Override  
  9.    public void doSomething()  
  10.    {  
  11.       super.doSomething();  
  12.        
  13.       this.doAnotherDosomething();  
  14.        
  15.    }  
  16.     
  17.    private void doAnotherDosomething()  
  18.    {  
  19.       System.out.println("功能C");  
  20.    }  
  21. }  

6)   客户端

[java]  view plain copy print ?
  1. public class Client {  
  2.     public static void main(String[] args) {  
  3.   
  4.         Component component = new ConcreteDecorate1(new ConcreteDecorate1(  
  5.                 new ConcreteComponent()));  
  6.         component.doSomething();  
  7.     }  
  8. }  

 

五 、装饰模式的特点:

1) 装饰对象和真实对象具有相同的接口,这样客户端对象就可以以真实对象的相同的方式和装饰对象交互。

2) 装饰对象包含一个真实对象的引用(reference).

3) 装饰对象接受所有来自客户端的请求,它把这些请求转发给真实的对象。

4) 装饰对象可以在转发这些请求以前或者以后增加一些附加的功能。这样就能确保在运行时,不用修改给定对象结构就可以在外部增加附加的功能。在面向对象的程序设计中,通常是使用继承的关系来扩展给定类的功能。

六、装饰模式与类继承的区别:

1)装饰模式是一种动态行为,对已经存在类进行随意组合,而类的继承是一种静态的行为,一个类定义成什么样的,该类的对象便具有什么样的功能,无法动态的改变。

2)装饰模式扩展的是对象的功能,不需要增加类的数量,而类继承扩展是类的功能,在继承的关系中,如果我们想增加一个对象的功能,我们只能通过继承关系,在子类中增加两个方法。

3)装饰模式是在不改变原类文件和使用继承的情况下,动态的扩展一个对象的功能,它是通过创建一个包装对象,也就是装饰来包裹真是的对象。

4)装饰模式把对客户端的调用委派给被装饰的类,装饰模式的关键在于这种扩展完全透明的。

七、使用装饰模式的优点和缺点 

点:

  装饰模式与继承关系的目的都是要扩展对象的功能,但是装饰模式可以提供比继承更多的灵活性。

  通过使用不同的具体装饰类以及这些装饰类的排列组合,设计师可以创造出很多不同行为的组合。

  这种比继承更加灵活机动的特性,也同时意味着装饰模式比继承更加易于出错。

缺点:

  由于使用装饰模式,可以比使用继承关系需要较少数目的类。使用较少的类,当然使设计比较易于进行。但是,在另一方面,使用装饰模式会产生比使用继承关系更多的对象。更多的对象会使得查错变得困难,特别是这些对象看上去都很相像

八、例子2  打印票据

http://www.blogjava.net/flustar/archive/2007/11/28/decorator.html

例如,我们要为超市的收银台设计一个打印票据的程序,有的需要打印票据的头信息,有的需要打印票据的页脚信息,有的只需要打印票据的内容。如果针对每一种情况都修改一次程序,势必会很麻烦。这时我们可以考虑使用Decorator模式。其结构类图如下:

 

Component

[java]  view plain copy print ?
  1. abstract class Component {  
  2.     abstract public void printTicket();  
  3. }  

SalesTicket

[java]  view plain copy print ?
  1. class SalesTicket extends Component {  
  2.     public void printTicket() {  
  3.         System.out.println("打印出salesTicket的内容");  
  4.     }  
  5. }  

TicketDecorator

[java]  view plain copy print ?
  1. abstract class TicketDecorator extends Component {  
  2.     private Component myTrailer;  
  3.   
  4.     public TicketDecorator(Component myComponent) {  
  5.         myTrailer = myComponent;  
  6.     }  
  7.   
  8.     public void callTrailer() {  
  9.         if (myTrailer != null)  
  10.             myTrailer.printTicket();  
  11.     }  
  12. }  

Header

[java]  view plain copy print ?
  1. class Header extends TicketDecorator {  
  2.     public Header(Component myComponent) {  
  3.         super(myComponent);  
  4.     }  
  5.   
  6.     public void printTicket() {  
  7.         System.out.println("打印salesTicket的头信息");  
  8.         super.callTrailer();  
  9.   
  10.     }  
  11. }  

Footer

[java]  view plain copy print ?
  1. class Footer extends TicketDecorator {  
  2.     public Footer(Component myComponent) {  
  3.         super(myComponent);  
  4.     }  
  5.   
  6.     public void printTicket() {  
  7.         super.callTrailer();  
  8.         System.out.println("打印salesTicket的页脚信息");  
  9.     }  
  10. }  

Client

[java]  view plain copy print ?
  1. public class Client {  
  2.   
  3.     public static void main(String[] args) {  
  4.         System.out.println("====================================");  
  5.         new Header(new Footer(new SalesTicket())).printTicket();  
  6.         System.out.println("====================================");  
  7.         new Footer(new Header(new SalesTicket())).printTicket();  
  8.         System.out.println("====================================");  
  9.     }  
  10.   
  11. }  

运行结果

====================================
打印salesTicket的头信息
打印出salesTicket的内容
打印salesTicket的页脚信息
====================================
打印salesTicket的头信息
打印出salesTicket的内容
打印salesTicket的页脚信息
====================================

 

九、java I/O中的装饰模式的应用

1)抽象构建角色(Component):给出一个抽象的接口,以规范准备接受附加责任的对象。相当于I/O流里面InputStream/OutputStreamReader/Writer

2)具体的构建角色(ConcreteComponent):定义一个将要接受附加责任的类。相当于I/O里面的FileOutputStreamFileInputStream

3)装饰角色(Docorator):持有一个抽象构建(Component)角色的引用,并定义一个与抽象构件一致的接口。相当于I/O里面的FilerOutputStreamFilterInputStream

4)具体的装饰角色(ConcreteDecorator):负责给构建对象“贴上”附加的责任。相当于I/O流里面的BufferedOutputStreamBufferedInputStream以及DataOutputStreamDataInputSrtream

参考 

设计模式之禅

http://www.cnblogs.com/wangjq/archive/2012/07/03/2574755.html

http://www.blogjava.net/flustar/archive/2007/11/28/decorator.html

http://www.51cto.com/specbook/11/11650.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值