设计原则
类应该对扩展开放,对修改关闭
简单的说,就是在不修改之前写过的代码的基础上对系统的功能进行扩展,现有的一些设计模式通过提供扩展的方法已经可以做到完全遵守开放-关闭原则。
设计模式
装饰者模式
动态地将责任附加到对象上。若要扩展功能,装饰者提供了比继承更有弹性的替代方案。
书上的例子可以简化如下:
某咖啡店需要做一个下订单的软件,该店提供低咖啡因(6元)和浓缩(7元)两种咖啡,客户还可以另外在上面添加“配料”,配料有牛奶(4元)和摩卡(5元)。
不使用装饰者模式的实现方法如下:
public class DecafWithMilk{
public int cost(){
return 4 + 6;
}
}
public class EspressoWithMilk{
public int cost(){
return 4 + 7;
}
}
public class DecafWithMocha{
public int cost(){
return 5 + 6;
}
}
public class EspressoWithMocha{
public int cost(){
return 5 + 7;
}
}
上面这种实现方式会导致下面的这几种问题:
1.万一有客人要加两份牛奶怎么办?万一有客人要加十份牛奶怎么办?
2.如果某天该店的店长想加入一个新的配料:奶泡。是不是又要新加至少2个类?
3.如果某天牛奶的价格上涨了,我们需要修改几个类?
下面用装饰者模式就可以一下解决上面三个问题!!!简直不要太棒!
首先,定一个被装饰者的抽象类:
public abstract class Coffee{
public abstract int cost();
}
然后是配料的抽象类:
public abstract class Condiment extends Coffee{
public abstract int cost();
}
下面是两种咖啡的实现:
public class Decaf extends Coffee{
public int cost(){
return 6;
}
}
public class Espresso extends Coffee{
public int cost(){
return 7;
}
}
下面是两种配料的实现:
public class AddMilk extends Condiment{
private Coffee coffee;
public AddMilk(Coffee coffee){
this.coffee = coffee;
}
public int cost(){
return 4 + coffee.cost();
}
}
public class AddMocha extends Condiment{
private Coffee coffee;
public AddMocha(Coffee coffee){
this.coffee = coffee;
}
public int cost(){
return 5 + coffee.cost();
}
}
类之间关系如图:
下面是一位特殊的客人点的加一份摩卡加三份牛奶的浓缩咖啡:
public static void main(Stirng args[]){
Coffee orderCoffee = new Espresso();
orderCoffee = new AddMocha(orderCoffee);
orderCoffee = new AddMilk(orderCoffee);
orderCoffee = new AddMilk(orderCoffee);
System.out.println("Cost:" + orderCoffee.cost());
}
如果新来了奶泡这种配料(8元),只需要新加一个类:
public class AddFoam extends Condiment{
private Coffee coffee;
public AddMocha(Coffee coffee){
this.coffee = coffee;
}
public int cost(){
return 8 + coffee.cost();
}
}
上面的客人就可以多加一份奶泡了:
public static void main(Stirng args[]){
Coffee orderCoffee = new Espresso();
orderCoffee = new AddMocha(orderCoffee);
orderCoffee = new AddMilk(orderCoffee);
orderCoffee = new AddMilk(orderCoffee);
orderCoffee = new AddFoam(orderCoffee);
System.out.println("Cost:" + orderCoffee.cost());
}