参考 : Head First 设计模式(中文版)
这篇只作为个人温习!
用意 : 动态地给一个对象添加|扩展一些行为。Decorator 强调用对象组合而非继承来实现扩展,这显得较为灵活。
角色:
1.装饰者
2.被装饰者
特点与原理 :
0."被装饰者"被装饰之后,会被"装饰者"给包起来,外界看来它是被替代掉了,或者说原本的"被装饰者"被隔离了。
1."装饰者 "和"被装饰者 "必须是同一个抽象类派生出来的。(因为"装饰者"必须能替代掉"被装饰者")
2."装饰者"也可以被装饰(可以一直被再装饰),但是"被装饰者"不能装饰别人.
3."被装饰者"可以说是一个起点,一但它被"装饰者"装饰后,装饰者将替代掉它.
4.比如 a 对象被 b对象装饰后 , 外界就不能在访问a对象,b对象会保留a对象指针,那么外界通过调用 b 对象的方法来实现功能 , 这样我们便可以定义扩展方法在b对象上,b既可以调用原始a对象的方法,也可以添加扩展方法。
简单例子 :
public abstract class Product //基类 { public abstract double getPrice(); //既然是要扩展,自然是用抽象方法咯 } public class Nasi : Product //"被装饰者"一定要派生自基类 { public override double getPrice() { return 100.00; } } public abstract class Decorator : Product //"装饰者"一定要派生自基类 { public Product product { get; set; } //每个装饰者都必须保留被装饰对象的指针 } public class Sayur : Decorator { public Sayur(Product product) //这里的product既可以是"装饰者"也可以是"被装饰者" { this.product = product; //保存指针 } public override double getPrice() { return this.product.getPrice() + 10; //调用原来方法,再添加扩展逻辑 } } protected void Page_Load(object sender, EventArgs e) { Product nasi = new Nasi(); Product nasiWithSayur = new Sayur(nasi); //用Sayur这个"装饰者"来装饰product,装饰后product就被替换掉了. Product nasiWithDoubleSayur = new Sayur(nasiWithSayur); //第2次装修返回的又是另一个对象了 double price = nasiWithDoubleSayur.getPrice();//由于我们是面向抽象编程,所以不管怎么装饰都好,这个对象依旧是一个Product,所以就一定有getPrice方法,只是它的内部已经是不同的实现了。 }
记忆联想 :
规范 : 装饰对象,和被装饰的对象都派生自一个类
实现 : 对象包对象,外层对象的方法是调用内层对象并且加上自己的逻辑扩展。