結構型模式:裝飾模式
文章首發:
結構型模式:裝飾模式
七大結構型模式之四:裝飾模式。
簡介
姓名 :裝飾模式
英文名 :Decorator Pattern
價值觀 :人靠衣裝,類靠裝飾
個人介紹 :
Attach additional responsibilities to an object dynamically keeping the same interface. Decorators provide a flexible alternative to subclassing for extending functionality.
動態地給一個對象添加一些額外的職責。就增加功能來說,裝飾模式相比生成子類更為靈活。
你要的故事
夏天到了,吃貨們期待的各種各樣冷凍零食就要大面積面向市場,什麽冰淇淋、雪糕、冰棍等等。今天的裝飾模式不講這些都受歡迎的零食,講講那烏黑滴龜苓膏。不知道大夥們喜不喜歡吃龜苓膏,我是挺喜歡的,不喜歡的人很多都閑它苦,應該沒有人願意在沒加任何糖類的情況下吃龜苓膏。很多糖水店會提供幾種龜苓膏,比如蜂蜜龜苓膏、牛奶龜苓膏。下面我們空想出一個場景來。
天氣到了 30℃,小明和小紅加班到了 10 點,一起下班,路過一家糖水店,小紅萌生了吃糖水解解熱的想法,小明最近漲薪,就提出要請小紅吃糖水,他們進去糖水店,小紅想著好久沒吃龜苓膏了,就想吃吃,懷念一下童年那段在農村夜晚吃龜苓膏的時光。糖水店裏面有 3 種龜苓膏,一種是普通龜苓膏(這老板居然提供不加任何糖分的,過分了),一種是蜂蜜龜苓膏,另外一種是牛奶龜苓膏,小明點了一份蜂蜜龜苓膏,小紅想加蜂蜜和牛奶,就咨詢了老板娘,能否同時加蜂蜜和牛奶,老板娘用那東北腔爽快地回復小紅:行。小明和小紅就等龜苓膏上桌。。。腦洞到這。
我們來把這個故事套入到裝飾模式裏去。上面故事裏老板賣 3 種龜苓膏,而都不滿足小紅的需求,小紅想要的是蜂蜜牛奶龜苓膏,如果用繼承來實現龜苓膏,那就無法滿足小紅的要求了,因為繼承直接固定了龜苓膏的做法,加什麽就是什麽,要蜂蜜牛奶龜苓膏,那就需要另外一個龜苓膏類代表蜂蜜牛奶龜苓膏;而用裝飾模式則不同,下面看看裝飾模式的實現代碼。
龜苓膏抽象類,該類定義了制作龜苓膏的抽象方法。
/**
* 龜苓膏
*/
abstract class HerbalJelly {
/**
* 制作龜苓膏方法
*/
public abstract void process();
}
老板提供的最基本的龜苓膏,這種龜苓膏不加任何料,就是那苦苦的龜苓膏,我們稱它為普通龜苓膏。
/**
* 普通龜苓膏
*/
class CommonHerbalJelly extends HerbalJelly {
@Override
public void process() {
System.out.println("盛一碗龜苓膏");
}
}
另外 2 種龜苓膏:蜂蜜龜苓膏和牛奶龜苓膏,不是用繼承實現,而是用裝飾器實現,我們可以發現這 2 種龜苓膏都是基於上面普通龜苓膏添加不同的糖類食品制作而成。下面實現一個抽象類充當裝飾器。
/**
* 龜苓膏裝飾器
*/
abstract class Decorator extends HerbalJelly {
private HerbalJelly herbalJelly;
public Decorator(HerbalJelly herbalJelly) {
this.herbalJelly = herbalJelly;
}
@Override
public void process() {
this.herbalJelly.process();
}
}
接下來就根據上面的龜苓膏裝飾器來實現蜂蜜龜苓膏和牛奶龜苓膏。
/**
* 蜂蜜龜苓膏
*/
class HoneyHerbalJelly extends Decorator{
public HoneyHerbalJelly(HerbalJelly herbalJelly) {
super(herbalJelly);
}
@Override
public void process() {
super.process();
System.out.println("加蜂蜜");
}
}
/**
* 牛奶龜苓膏
*/
class MilkHerbalJelly extends Decorator{
public MilkHerbalJelly(HerbalJelly herbalJelly) {
super(herbalJelly);
}
@Override
public void process() {
super.process();
System.out.println("加牛奶");
}
}
下面提供我們的測試代碼,還記得上面說的,小明要了一碗蜂蜜龜苓膏,小紅則要了一碗蜂蜜牛奶龜苓膏。
public class DecoratorTest {
public static void main(String[] args) {
CommonHerbalJelly commonHerbalJelly = new CommonHerbalJelly();
HoneyHerbalJelly honeyHerbalJelly = new HoneyHerbalJelly(commonHerbalJelly);
// 小明的蜂蜜龜苓膏
honeyHerbalJelly.process();
MilkHerbalJelly milkHerbalJelly = new MilkHerbalJelly(honeyHerbalJelly);
// 小紅的蜂蜜牛奶龜苓膏
milkHerbalJelly.process();
}
}
打印結果:
盛一碗龜苓膏
加蜂蜜
盛一碗龜苓膏
加蜂蜜
加牛奶
我們看到,小明的龜苓膏只加蜂蜜,小紅的龜苓膏加了蜂蜜和牛奶,這樣就很簡單的滿足了小紅的要求。
總結
裝飾模式在一些類與類之間有疊加效應(也就是給一個類增加附加功能)的場景中非常好用,它可以說是繼承的替代品,有更好的擴展性,也比較靈活。在 Java JDK 源碼中也大面積用到了裝飾模式,比如:java.io.BufferedInputStream(InputStream)。學完基礎知識後,可以看看源碼中是怎麽實現的,鞏固知識。
推薦閱讀
結構型模式:組合模式
結構型模式:橋接模式
結構型模式:適配器模式
公眾號後臺回復『大禮包』獲取 Java、Python、IOS 等教程
加個人微信備註『教程』獲取架構師、機器學習等教程
結構型模式:裝飾模式