装饰者模式
1、定义:动态的给一个对象添加一些额外的职责,就增加功能来说,装饰模式比生成子类更为灵活。
2、理解:子类扩展毕竟是静态的方案,由于是静态的方案毕竟没有动态的扩展灵活,所以对于动态扩展来说,要灵活很多,这或许也是装饰者模式的一个重要体现。
3、开发的观念:我们所设计的类要对扩展开放,对修改关闭。当我们设计的类不能满足我们的需求的时候,我们可能设计一个类去继承它,但是这样就会使对象之间高度的耦合,解耦的时候,或者说需要维护的时候会造成很多不必要的麻烦,这个时候,我们就可以考虑一下使用装饰者模式了,装饰者模式简单来说,就是把对象嵌入我们要扩展功能的类中,调用他的方法,然后跟我们定义的方法一起返回我们需要的类型。这个可能跟适配器模式有点相似,但是他们所针对的点不一样。
4、装饰者模式UML图:
5、例子:当我们去咖啡厅的时候。不知道是不是会遇到这种问题呢?本来你点一杯普通的coffe,然后他的价格是10元,但是你尝了之后发现有点苦,想要加点红糖,然后加这个糖的价格是2元;然后过了一会你的朋友过来了,他叫了一杯普通coffee,然后加了点牛奶(假设可以加),这个牛奶的价格是3元,如果叫你设计一个程序负责这部分的内容,你会怎么做呢?
会不会有人这么想,java不是面向对象的语言吗?我们可以通过设计对象来啊,首先创建coffee一个抽象类,然后在列出可能添加调料的可能食品,然后付款的时候,调用方法不就行了吗?确实这样做是可以的,但是前面说了耦合度高,这个时候,就到了我们的装饰者出场了:
关系图:
我们要做的就是点咖啡1的时候加sugar,点咖啡2的时候加milk
(1)创建coffee的接口
package com.zqu.yqy.scdn.test.test006; public abstract class coffee { String coffeeInformation = "普通咖啡"; public String getCoffeeInformation(){ return coffeeInformation; } public abstract double cost(); }
实现类coffee1,coffee2:
实现coffee1:
package com.zqu.yqy.scdn.test.test006; public class coffee1 extends coffee { public coffee1(){ coffeeInformation = "coffee1"; } @Override public double cost() { return 10.0; } }
实现coffee2:
package com.zqu.yqy.scdn.test.test006; public class coffee2 extends coffee { public coffee2(){ coffeeInformation = "coffee2"; } @Override public double cost() { return 12.0; } }
(2)调料类
package com.zqu.yqy.scdn.test.test006; public abstract class AddThings extends coffee { public abstract String getCoffeeInformation(); }
实现类:
coffee1添加牛奶:
package com.zqu.yqy.scdn.test.test006; public class milk extends AddThings{ coffee coff; public milk(coffee c) { this.coff = c; } @Override public String getCoffeeInformation() { String addThings = coff.getCoffeeInformation()+"添加"+"milk"; return addThings; } @Override public double cost() { return 3.0+coff.cost(); } }
coffee2添加糖:
package com.zqu.yqy.scdn.test.test006; public class sugar extends AddThings { coffee coff; public sugar(coffee c) { this.coff = c; } @Override public String getCoffeeInformation() { String addThings = coff.getCoffeeInformation()+"添加"+"sugar"; return addThings; } @Override public double cost() { return 2.0+coff.cost(); } }
测试类:
package com.zqu.yqy.scdn.test.test006; public class cost { public static void main(String[] args) { coffee c1 = new coffee1(); coffee c2 = new coffee2(); coffee a1 = new sugar(c1); coffee a2 = new milk(c1); //coffee a3 = new milk(c2); System.out.println(a1.getCoffeeInformation()+" "+a1.cost()); System.out.println(a2.getCoffeeInformation()+" "+a2.cost()); //System.out.println(a3.getCoffeeInformation()+" "+a3.cost()); } }
结果: