1. 程式人生 > >結構型模式:裝飾模式

結構型模式:裝飾模式

結果 什麽 inpu 不同的 滿足 裝飾模式 ebr pro 對象

文章首發:
結構型模式:裝飾模式

技術分享圖片

七大結構型模式之四:裝飾模式。

簡介

姓名 :裝飾模式

英文名 :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 等教程
加個人微信備註『教程』獲取架構師、機器學習等教程

技術分享圖片

結構型模式:裝飾模式