锈才学设计模式之 —— 装饰者模式(Decorate Pattern)

锈才学设计模式之  —— 装饰者模式 

 

装饰者模式:在运行时动态的将行为扩展到装饰者对象上,符合"对扩展开放,对修改关闭"原则.

 

说明:

       通俗点说,装饰者模式就是,一个类把另一个类装载进来进行包装,实现更多特性,他们都实现相同的接口.比如:

       形像设计店,有顾客进去,设计师就可以把顾客造型,包装成一个新的形象.

       装修公司,接手一个毛坏房,设计师就可以把房子装饰成一个风格时尚的精品房

 

       在设计一个扩展的功能时,一般都是使用继承,由子类继承父类.比如商品对象:

这种继承关系的设计是对功能提供了扩展,但是设计的弹性不够.

比如:

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个葫芦娃变成一个葫芦娃时,这个葫芦娃就是装饰者,其它葫芦娃是被装饰者,装饰者集中了大力士、千里眼和顺风耳、硬铁头、火功、水性、隐身术、宝葫芦功能强大。所以能胜利。

 

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值