decorator
閱讀原文請(qǐng)?jiān)L問(wèn)我的博客BrightLoong's Blog
一. 概述
裝飾器模式(Decorator)
,動(dòng)態(tài)地給一個(gè)對(duì)象添加一些額外的職責(zé)设褐,就增加功能來(lái)說(shuō)律罢,裝飾器模式比生成子類更為靈活高帖;它允許向一個(gè)現(xiàn)有的對(duì)象添加新的功能找御,同時(shí)又不改變其結(jié)構(gòu)。
裝飾器模式屬于結(jié)構(gòu)型模式乳丰。
二. UML類圖解析
裝飾器模式的UML類圖如下:
decorator
- Component:接口掌测,定義一個(gè)抽象接口,真實(shí)對(duì)象和裝飾對(duì)象具有相同的接口产园,以便動(dòng)態(tài)的添加職責(zé)汞斧。
- ConcreteComponent:具體的對(duì)象。
- Decorator:裝飾類淆两,繼承了Component,從外類來(lái)擴(kuò)展Component類的功能断箫,并且持有一個(gè)構(gòu)建引用拂酣,進(jìn)行請(qǐng)求轉(zhuǎn)發(fā)秋冰。
- ConcreteDecorator:具體裝飾類,用于給實(shí)際對(duì)象添加職責(zé)婶熬。
三. 代碼實(shí)現(xiàn)
現(xiàn)在考慮這樣一個(gè)場(chǎng)景剑勾,現(xiàn)在有一個(gè)煎餅攤埃撵,人們?nèi)ベI煎餅(Pancake),有些人要加火腿(Ham)的,有些人要加雞蛋(Egg)的虽另,有些人要加生菜(Lettuce)的暂刘,當(dāng)然土豪可能有啥全給加上_。用上述的裝飾器模式來(lái)進(jìn)行編碼捂刺。
1. 定義煎餅接口IPancake
package io.github.brightloong.design.decorator;
/**
* 定義一個(gè)煎餅接口
* Created by BrightLoong on 2018/4/22.
*/
public interface IPancake {
/**
* 定義烹飪的操作
*/
void cook();
}
2. 定義具體的煎餅Pancake
package io.github.brightloong.design.decorator;
/**
* 具體的煎餅對(duì)象谣拣,可用其他裝飾類進(jìn)行動(dòng)態(tài)擴(kuò)展。
* Created by BrightLoong on 2018/4/22.
*/
public class Pancake implements IPancake{
public void cook() {
System.out.println("的煎餅");
}
}
3. 定義抽象裝飾類PancakeDecorator
package io.github.brightloong.design.decorator;
/**
* 實(shí)現(xiàn)接口的抽象裝飾類族展,建議設(shè)置成abstract.
* Created by BrightLoong on 2018/4/22.
*/
public abstract class PancakeDecorator implements IPancake {
/***/
private IPancake pancake;
public PancakeDecorator(IPancake pancake) {
this.pancake = pancake;
}
public void cook() {
if (this.pancake != null) {
pancake.cook();
}
}
}
4. 具體裝飾類EggDecorator
package io.github.brightloong.design.decorator;
/**
* 對(duì)煎餅加雞蛋的裝飾類森缠,繼承PancakeDecorator,覆蓋cook操作
* Created by BrightLoong on 2018/4/22.
*/
public class EggDecorator extends PancakeDecorator {
public EggDecorator(IPancake pancake) {
super(pancake);
}
/**
* 覆蓋cook方法仪缸,加入自身的實(shí)現(xiàn)贵涵,并且調(diào)用父類的cook方法,也就是構(gòu)造函數(shù)中
* EggDecorator(IPancake pancake)恰画,這里傳入的pancake的cook操作
*/
@Override
public void cook() {
System.out.println("加了一個(gè)雞蛋宾茂,");
super.cook();
}
}
5. 具體裝飾類HamDecorator
package io.github.brightloong.design.decorator;
/**
* 對(duì)煎餅加火腿的裝飾類,繼承PancakeDecorator拴还,覆蓋cook操作
* Created by BrightLoong on 2018/4/22.
*/
public class HamDecorator extends PancakeDecorator {
public HamDecorator(IPancake pancake) {
super(pancake);
}
/**
* 覆蓋cook方法跨晴,加入自身的實(shí)現(xiàn),并且調(diào)用父類的cook方法自沧,也就是構(gòu)造函數(shù)中
* EggDecorator(IPancake pancake)坟奥,這里傳入的pancake的cook操作
*/
@Override
public void cook() {
System.out.println("加了一根火腿,");
super.cook();
}
}
6. 具體裝飾類LettuceDecorator
package io.github.brightloong.design.decorator;
/**
* 對(duì)煎餅加生菜的裝飾類拇厢,繼承PancakeDecorator爱谁,覆蓋cook操作
* Created by BrightLoong on 2018/4/22.
*/
public class LettuceDecorator extends PancakeDecorator {
public LettuceDecorator(IPancake pancake) {
super(pancake);
}
/**
* 覆蓋cook方法,加入自身的實(shí)現(xiàn)孝偎,并且調(diào)用父類的cook方法访敌,也就是構(gòu)造函數(shù)中
* EggDecorator(IPancake pancake),這里傳入的pancake的cook操作
*/
@Override
public void cook() {
System.out.println("加了一顆生菜衣盾,");
super.cook();
}
}
7. 客戶端調(diào)用以及結(jié)果
package io.github.brightloong.design.decorator;
/**
* 調(diào)用客戶端
* Created by BrightLoong on 2018/4/22.
*/
public class Client {
public static void main(String[] args) {
System.out.println("=========我是土豪都給我加上===========");
IPancake pancake = new Pancake();
IPancake pancakeWithEgg = new EggDecorator(pancake);
IPancake pancakeWithEggAndHam = new HamDecorator(pancakeWithEgg);
IPancake panckeWithEggAndHamAndLettuce = new LettuceDecorator(pancakeWithEggAndHam);
panckeWithEggAndHamAndLettuce.cook();
System.out.println("==========我是程序猿寺旺,加兩個(gè)雞蛋補(bǔ)補(bǔ)==============");
IPancake pancake2 = new Pancake();
IPancake pancakeWithEgg2 = new EggDecorator(pancake2);
IPancake pancakeWithTwoEgg = new EggDecorator(pancakeWithEgg2);
pancakeWithTwoEgg.cook();
}
}
輸出結(jié)果
=========我是土豪都給我加上===========
加了一顆生菜,
加了一根火腿势决,
加了一個(gè)雞蛋阻塑,
的煎餅
==========我是程序猿,加兩個(gè)雞蛋補(bǔ)補(bǔ)==============
加了一個(gè)雞蛋果复,
加了一個(gè)雞蛋陈莽,
的煎餅
四. 總結(jié)
關(guān)于裝飾器模式的使用塞绿,在我看來(lái)主要有一下幾點(diǎn)需要注意的
- 抽象裝飾器和具體被裝飾的對(duì)象實(shí)現(xiàn)同一個(gè)接口
- 抽象裝飾器里面要持有接口對(duì)象坪蚁,以便請(qǐng)求傳遞
- 具體裝飾器覆蓋抽象裝飾器方法并用super進(jìn)行調(diào)用畏纲,傳遞請(qǐng)求
1. 適用場(chǎng)景
- 擴(kuò)展一個(gè)類的功能筛武。
- 動(dòng)態(tài)添加功能,動(dòng)態(tài)撤銷私植。
2. 優(yōu)點(diǎn)
- 裝飾類和被裝飾類都只關(guān)心自身的核心業(yè)務(wù)忌栅,實(shí)現(xiàn)了解耦曲稼。
- 方便動(dòng)態(tài)的擴(kuò)展功能,且提供了比繼承更多的靈活性贫悄。
3. 缺點(diǎn)
- 如果功能擴(kuò)展過(guò)多者春,勢(shì)必產(chǎn)生大量的類清女。
- 多層裝飾比較復(fù)雜。