基本定义
- 装饰者模式:动态的将新功能附加到对象上。 在对象功能扩展方面,它比继承更具有弹性,装饰者模式也体现了开闭原则。
- 这里提到的动态的将新功能附加到对象和ocp原则,在后面的应用实例上会以代码的形式展现。
装饰者模式原理
- 装饰者模式在星巴克咖啡的比喻如下所示:
- 主体:比如——各式各样的咖啡
- 包装:比如——各种各样的调料
- Component主体:比如类似前面的Drink
- ConcreteComponent和DecoratorConcreteComponent:具体的主体,比如前面的各个单品咖啡
- Decorator:装饰者,比如各调料.在如图的Component与ConcreteComponent之间,如果ConcreteComponent类很多,还可以设计一个缓冲层,将共有的部分提取出来,抽象层一个类。
UML类图如图所示
- 装饰者模式下的订单:2份巧克力+1份牛奶的LongBlack
代码实现
定义一个Drink类(饮料类的抽象)
public abstract class Drink {
public String des;
private float price = 0.0f;
public String getDes() {
return des;
}
public void setDes(String des) {
this.des = des;
}
public float getPrice() {
return price;
}
public void setPrice(float price) {
this.price = price;
}
public abstract float cost();
}
定义一个咖啡类,它是对咖啡的一个抽象,也是介于各种咖啡和饮料之间的一个缓冲层。
public class Coffee extends Drink {
@Override
public float cost() {
return super.getPrice();
}
}
定义各种各样的具体咖啡类
public class LongBlack extends Coffee {
public LongBlack() {
setDes(" longblack ");
setPrice(5.0f);
}
}
public class ShortBlack extends Coffee{
public ShortBlack() {
setDes(" shortblack ");
setPrice(4.0f);
}
}
public class Espresso extends Coffee {
public Espresso() {
setDes(" 意大利咖啡 ");
setPrice(6.0f);
}
}
定义调料品抽象类
public class Decorator extends Drink {
private Drink obj;
public Decorator(Drink obj) {
this.obj = obj;
}
@Override
public float cost() {
return super.getPrice() + obj.cost();
}
@Override
public String getDes() {
return des + " " + getPrice() + " && " + obj.getDes();
}
}
各种调料的具体实现类
public class Chocolate extends Decorator {
public Chocolate(Drink obj) {
super(obj);
setDes(" 巧克力 ");
setPrice(3.0f);
}
}
public class Milk extends Decorator {
public Milk(Drink obj) {
super(obj);
setDes(" 牛奶 ");
setPrice(2.0f);
}
}
public class Soy extends Decorator{
public Soy(Drink obj) {
super(obj);
setDes(" 豆浆 ");
setPrice(1.5f);
}
}
测试类,及其效果展示
public class CoffeeBar {
public static void main(String[] args) {
Drink order = new LongBlack();
System.out.println("费用1=" + order.cost());
System.out.println("描述=" + order.getDes());
order = new Milk(order);
System.out.println("order 加入一份牛奶 费用 =" + order.cost());
System.out.println("order 加入一份牛奶 描述 = " + order.getDes());
order = new Chocolate(order);
System.out.println("order 加入一份牛奶 加入一份巧克力 费用 =" + order.cost());
System.out.println("order 加入一份牛奶 加入一份巧克力 描述 = " + order.getDes());
order = new Chocolate(order);
System.out.println("order 加入一份牛奶 加入2份巧克力 费用 =" + order.cost());
System.out.println("order 加入一份牛奶 加入2份巧克力 描述 = " + order.getDes());
System.out.println("===========================");
Drink order2 = new DeCaf();
System.out.println("order2 无因咖啡 费用 =" + order2.cost());
System.out.println("order2 无因咖啡 描述 = " + order2.getDes());
order2 = new Milk(order2);
System.out.println("order2 无因咖啡 加入一份牛奶 费用 =" + order2.cost());
System.out.println("order2 无因咖啡 加入一份牛奶 描述 = " + order2.getDes());
}
}
运行结果