介绍 :
- 动态扩展对象功能 相比继承而言更加灵活 可以替代继承 其装饰者是依赖于被装饰者而存在的
- 注意装饰者和被装饰者实现一个共同的接口 目的是为了保证其类型一致
- 优点 : 解耦了装饰者和被装饰者的关系 易扩展类型功能
- 缺点 : 拥有非常多的小类<被装饰者> 增加系统复杂度
以一个咖啡馆为例
-
在咖啡馆有牛奶和咖啡2种饮料<可以被看成是被装饰者> 还有配料可以加糖 蜂蜜 柠檬
-
如此 有的客户需要咖啡加糖 有的需要牛奶加蜂蜜加糖等等 如果使用继承关系来进行搭配 就会生成很多子类 而且子类会随着装饰者和被装饰者的增加而越来越多 最终导致子类膨胀且不易扩展
-
所以我们可以使用装饰者模式来替代继承关系 在后期咖啡店有了新的饮品和调料时 可以更加轻松的进行代码扩展 代码如下
-
接口
public interface Drinks {
String describe();//描述
Integer price();//价格
}
- 被装饰者类
public class Coffer implements Drinks {
@Override
public String describe() {
return "咖啡";
}
@Override
public Integer price() {
return 5;
}
}
======================
public class Milk implements Drinks {
@Override
public String describe() {
return "牛奶";
}
@Override
public Integer price() {
return 3;
}
}
- 装饰者类
public abstract class Condiments implements Drinks {
private Drinks drinks;
public Condiments() {
}
public Condiments(Drinks drinks) {
this.drinks = drinks;
}
//获取装饰实例对象数据
@Override
public String describe() {
return drinks.describe();
}
//获取装饰实例对象数据
@Override
public Integer price() {
return drinks.price();
}
}
========================
public class Honey extends Condiments{
public Honey(Drinks drinks) {
super(drinks);
}
@Override
public String describe() {
return super.describe() + ", 加蜂蜜";
}
@Override
public Integer price() {
return super.price() + 4;
}
}
========================
public class Sugar extends Condiments{
public Sugar(Drinks drinks) {
super(drinks);
}
@Override
public String describe() {
return super.describe() + ",加糖";
}
@Override
public Integer price() {
return super.price() + 2;
}
}
==========================
public class Lemon extends Condiments{
public Lemon(Drinks drinks) {
super(drinks);
}
@Override
public String describe() {
return super.describe() + ",加柠檬";
}
@Override
public Integer price() {
return super.price() + 3;
}
}
- 测试
public class Test {
public static void main(String[] args) {
Drinks coffer = new Coffer();
coffer = new Honey(coffer);
coffer = new Sugar(coffer);
System.out.println("饮品 : " + coffer.describe() + "===价格 : " + coffer.price());
Drinks milk = new Milk();
milk = new Lemon(milk);
System.out.println("饮品 : " + milk.describe() + "===价格 : " + milk.price());
}
}