锈才学设计模式之 —— 装饰者模式
装饰者模式:在运行时动态的将行为扩展到装饰者对象上,符合"对扩展开放,对修改关闭"原则.
说明:
通俗点说,装饰者模式就是,一个类把另一个类装载进来进行包装,实现更多特性,他们都实现相同的接口.比如:
形像设计店,有顾客进去,设计师就可以把顾客造型,包装成一个新的形象.
装修公司,接手一个毛坏房,设计师就可以把房子装饰成一个风格时尚的精品房
在设计一个扩展的功能时,一般都是使用继承,由子类继承父类.比如商品对象:
这种继承关系的设计是对功能提供了扩展,但是设计的弹性不够.
比如:
1. 新增了礼包和赠送商品,在暑假大促销活动中,当用户购买了实物商品配送大礼包,购买虚拟商品赠送虚似商品(如游戏装配),在这种情况,商品需要与礼包和赠送商品组合,就不得不重新修改商品的购买规则代码才能满足需求.如果活动取消,又要修改购买规则代码.
2. 其次就是,实物商品有很多种类:图书商品、电子商品、票务商品、食品商品等,礼包也包括:光碟、消费券等,如果具备好的设计,面对众多的业务规则变化,扩展和维护都将是一个非常烦恼的事。
我们需要一种面向接口设计、对象之间松耦合的模式,来解决问题。
这里学习的装饰者模式,就提供了一种更具弹性的扩展形式.扩展的时候不需要修改原有的类,就能将新的业务行为增加到原有的对象中.同是做到“对扩展开放,对修改关闭”的OO设计原则.装饰者模式的结构图:
根据装饰者模式,将商品的设计做修改。如下图:
经过装饰者模式的设计后,我们就解决了上述的问题:
礼包和赠送商品都继承了商品抽象类(装饰者接口,这里也可以是实现接口,因为商品是继承商品基类,所以也就采用抽象类),同时礼包和赠送商品中都包括了一个商品基类的引用(如果是接口,就是接口的引用),达到类型一致,面向接口编程。当开展暑假促销活动时,用户购买了一个实物商品(比如一本书),我们就配送一份礼包(比如:购书券).代码如下:
示例代码:
== 商品基类 ==
package decorator; //商品基类 public abstract class Merchandise { private String merchandiseId; private String merchandiseName; private Double price; private int stock; //购买 public void buy(){ //具体购买规则 } }
== 实物商品(图片)类 ==
package decorator; public class BookMerchandise extends Merchandise{ //复写购买 public void buy(){ } }
== 装饰抽象类 ==
package decorator; //装饰抽象类或接口 public abstract class Decorator extends Merchandise{ //继承上面的方法 public void buy(){ } //可以实现一些其它方法 public abstract String showDescription(); }
== 礼包类 ==
package decorator; //礼包或赠送商品 public class Gift extends Decorator{ Merchandise merchandise;//这里是商品基类,也就是被装饰者的基类,在构造器中赋值 public Gift(Merchandise merchandise){ this.merchandise = merchandise; } @Override public String showDescription() { // TODO Auto-generated method stub return null; } @Override public void buy() { // 购买行为,可以包括被装饰者的行为如图书商品的购买 merchandise.buy();//被装饰者的购买 //礼品的配送(购买) } }
== 测试购买商品类 ==
package decorator; //测试购买 public class BuyTest { public static void main(String[] args){ //直接购买图书商品, Merchandise merchandise = new BookMerchandise(); merchandise.buy(); //----------------------------------------------- //购买图书商品配送礼包 Merchandise merchandise1 = new BookMerchandise(); //将图书商品(merchandise)装饰到Gift(礼包)商品中 merchandise1 = new Gift(merchandise1); //调用礼包的购买(buy)方法,在方法中购买商品的时候配送礼包,或其它行为 merchandise.buy(); //当购买商品配送其它礼包,可以新增一个商品, //实现装饰者接口(Decorator)再将商品装饰进来就可以了 //这样以后增加商品或增加行为只要扩展对象就行,不需要修改原来的类. } }
上面的代码中,做到了非常有弹性的扩展性。增加商品或礼包等,都不会影响原有的类。扩展类的行为,只要实现装饰者接口或抽象类,就可以松耦合的增加类,增加行为。
在JDK AIP中,有很多地方实现装饰者模式。可以查看JDK查看内置的装饰者模式代码:java.io.InputStream,API结构如图:
这种设计模式在古时候劳动人民的智慧中就有体现,比如大家都看过<葫芦兄弟>这部动画片吧.葫芦兄弟,有红、橙、黄、绿、青、蓝、紫七个大葫芦娃。每个人都有独特的本领,红娃是大力士,橙娃是千里眼和顺风耳,黄娃是硬铁头,绿娃会火功,青娃有水性,蓝娃有隐身术,紫娃有宝葫芦,但是都斗不过蝎子精和蛇精,只有7个葫芦娃团结一起,集7个人的力量于一身时,才能发挥无穷能量,打败蝎子精和蛇精。这其实也是装饰者模式的例子,7个葫芦娃变成一个葫芦娃时,这个葫芦娃就是装饰者,其它葫芦娃是被装饰者,装饰者集中了大力士、千里眼和顺风耳、硬铁头、火功、水性、隐身术、宝葫芦功能强大。所以能胜利。