装饰器模式作用是针对目标方法进行增强,提供新的功能或者额外的功能。
不同于适配器模式和桥接模式,装饰器模式涉及的是单方,和代理模式相同,而且目标必须是抽象的。
而实际上,装饰器模式和代理模式的实现方式基本一致,只在目标的存在上有些差别,这个后面我们具体讲述。
上面提到了两点:
-
<李>涉及的是单方李>
<李>目标是抽象的李>
我们来想一下,所谓单方主要指的是在整个装饰器模式中不存在双方调用,要解决的也不是双方调用的问题,而是解决单方提供对外服务的问题,这个单方在自行对外提供服务时,功能不足,或者我们需要额外添加一些新功能,这时就可以使用装饰器模式,来对这个单方进行增强。
目标抽象的意思是因为我们需要通过实现接口的方式来进行增强,因此目标必须抽象为接口。
下面我们用我们生活中的一个例子来说明,我们用房子来作为目标:
房子接口:房子
/* * *目标接口:房子 */公共接口房子{ 无效的输出(); }
,具体的房子:DonghaoHouse
/* * *房子实现类 */公共类DonghaoHouse实现房子{ @Override 公共空间输出(){ System.out.println(“这是董浩的房子”); } }
具体的房子:DongliangHouse
/* * *房子实现类 */公共类DongliangHouse实现房子{ @Override 公共空间输出(){ System.out.println(“这是董量的房子”); } }
装饰器:装饰
公共类装饰器实现房子{ 私人住宅房子; 公共装饰(房子房子){ 这一点。房子=房子; } @Override 公共空间输出(){ System.out.println(“这是针对房子的前段装饰增强”); house.output (); System.out.println(“这是针对房子的后段装饰增强”); } }
测试类:
公共类客户{ 公共静态void main (String [] args) { 房子donghaoHouse=new donghaoHouse (); 房子装饰=new装饰(donghaoHouse); decorator.output (); } }
,执行结果为:
,这是针对房子的前段装饰增强
引用>
这是董浩的房子
这是针对房子的后段装饰增强
通过上面的例子我们可以看的出,除了测试类外,只剩下接口和实现类了,即使是装饰器类也是目标接口的一个字类,这更能说明单方的说法,模式中所有的类都属于目标方。至于目标是抽象的更是如此,只有目标是抽象的,才可以使用装饰器模式来进行增强。
上面我们说过装饰器模式与代理模式基本相同,只存在少许差别。
我们需要从概念上了解代理和装饰的区别:
<李>代理是全权代理,目标根本不对外,全部由代理类来完成。李> <李>装饰是增强,是辅助,目标仍然可以自行对外提供服务,装饰器只起增强作用。李>
上面两点提现到代码实现中是这样的:
代理模式公共类代理实现房子{ 私人住宅房子; 公共装饰(){ 这一点。房子=new DonghaoHouse (); } @Override 公共空间输出(){ System.out.println(“这是针对目标的前段增强”); house.output (); System.out.println(“这是针对目标的后段增强”); } },装饰模式
公共类装饰器实现房子{ 私人住宅房子; 公共装饰(房子房子){ 这一点。房子=房子; } @Override 公共空间输出(){ System.out.println(“这是针对房子的前段装饰增强”); house.output (); System.out.println(“这是针对房子的后段装饰增强”); } }看出来了吗,装饰器中持有的目标实例是从构造器传入的,而代理中持有的目标实例是自己创建的。
那么这里又出现一个区别,代理模式和装饰器模式虽然都依赖于目标接口,但是代理针对的目标实现类是固定的,而装饰器模式可以随意指定,也就是说目标是可以自有扩展的。
浅谈Java设计模式系列——装饰器模式