裝飾者模式可以動態地給一個對象添加一些額外的職責。就增加功能來說,Decorator模式相比生成子類更為靈活。
該模式的適用環境為:
(1)在不影響其他對象的情況下,以動態、透明的方式給單個對象添加職責。
(2)處理那些可以撤消的職責。
(3)當不能采用生成子類的方法進行擴充時。一種情況是,可能有大量獨立的擴展,為支持每一種組合將產生大量的子類,使得子類數目呈爆炸性增長。另一種情況可能是因為類定義被隱藏,或類定義不能用於生成子類。
實現該模式的關鍵步驟:
(1)Component(被裝飾對象基類):定義對象的接口,可以給這些對象動態增加職責;
(2)ConcreteComponent(具體被裝飾對象):定義具體的對象,Decorator可以給它增加額外的職責;
(3)Decorator(裝飾者抽象類):維護指向Component實例的引用,定義與Component一致的接口(也就是要繼承或實現被裝飾對象基類);
(4)ConcreteDecorator(具體裝飾者):具體的裝飾對象,給內部持有的具體被裝飾對象增加具體的職責;
這樣講大家可能有些不好理解,那我們還是老規矩:
入冬以後天氣越來越冷了,下班之後,做為資深吃貨,約上二三好友痛快的來場火鍋盛宴再爽不過了。說到火鍋,不得不提在成都吃過的大龍燚火鍋,各種鍋底,配菜應有盡有,但我最喜歡的還是大龍燚火鍋原味鍋底、麻辣牛肉、大刀毛肚、天味香腸、砣砣牛肉、麻辣排骨等,想想都流口水啊。
說道這大家結合裝飾者的實現步驟,應該有點感覺了吧,上文提到的鍋底,其實就是被裝飾對象的基類,配料其實就是裝飾者抽象類,大龍燚火鍋原味鍋底這些具體的鍋底也就是具體的被裝飾對象了,麻辣牛肉、大刀毛肚、天味香腸、砣砣牛肉、麻辣排骨這些裝飾鍋底用的各種配菜也就是具體的裝飾對象。說道這,大家應該都豁然開朗了吧,下面我們開始具體的代碼實現:
第一步:定義被裝飾對象基類(可以是抽象類也可以是接口)
1 public interface GuoDi { 2 public float cost();//鍋底當然要有價錢啦 3 public String name();//名字也得有哦 4 }
第二步:定義具體被裝飾對象(也就是各種鍋底,這裡定義兩種)
1 public class YuanYang implements GuoDi { 2 @Override 3 public float cost() { 4 return 48.0f; 5 } 6 @Override 7 public String name() { 8 return "鴛鴦鍋底"; 9 } 10 } 11 public class DaLongYan implements GuoDi{ 12 @Override 13 public float cost() { 14 return 59.0f; 15 } 16 @Override 17 public String name() { 18 return "大龍燚火鍋原味鍋底"; 19 } 20 }
第三步:定義裝飾者抽象類
1 public abstract class PeiCai implements GuoDi { 2 private GuoDi guodi; 3 public FoodDecorator(GuoDi guodi) { 4 super(); 5 this.guodi = guodi; 6 } 7 @Override 8 public float cost() { 9 return guodi.cost(); 10 } 11 @Override 12 public String name() { 13 return guodi.name(); 14 } 15 }
第四步:定義具體的裝飾者對象
1 public class MaLaNiuRou extends PeiCai { 2 public MaLaNiuRou(GuoDi guodi) { 3 super(guodi); 4 } 5 @Override 6 public float cost() { 7 return super.cost()+46f; 8 } 9 @Override 10 public String name() { 11 return super.name()+"+麻辣牛肉"; 12 } 13 } 14 public class MaoDu extends PeiCai { 15 16 public MaoDu(GuoDi guodi) { 17 super(guodi); 18 } 19 @Override 20 public float cost() { 21 return super.cost()+30f; 22 } 23 @Override 24 public String name() { 25 return super.name()+"+大刀毛肚"; 26 } 27 }
測試類:
1 public class Test { 2 public static void main(String[] args) { 3 GuoDi guodi = new DaLongYan ();//點個大龍燚火鍋原味鍋底 4 MaLaNiuRou y = new MaLaNiuRou(guodi);//來個麻辣牛肉 5 MaoDu x = new MaoDu(y);//在麻辣牛肉的基礎上再來個大刀毛肚 6 System.out.println("一共點了"+x.name()+",共消費"+s.cost()); 7 } 8 }
輸出結果:
1 一共點了大龍燚火鍋原味鍋底+麻辣牛肉+大刀毛肚,共消費135
不知道大家這頓吃的滿意不,不行的話,咱去成都來頓真正的裝飾者設計模式,看我不把我的大龍燚火鍋原味鍋底裝飾的讓你走不動道。