java設計模式-裝飾器模式(Decorator)

定義

裝飾器模式又名包裝(Wrapper)模式。裝飾器模式以對客戶端透明的方式拓展對象的功能,是繼承關系的一種替代方案。

裝飾器模式的結構

裝飾器模式以對客戶透明的方式動態的給一個對象附加上更多的責任。換言之,客戶端并不會覺得對象在裝飾前和裝飾后有什么不同。裝飾器模式可以在不是用創造更多子類的情況下,將對象的功能加以拓展。

裝飾器模式的類圖如下:

裝飾器模式的類圖

在裝飾器模式中的角色有:

  • 抽象構件(Component)角色:給出一個抽象接口,已規范準備接收附加責任的對象。
  • 具體構件(ConcreteComponent)角色:定義一個將要接收附加責任的類
  • 裝飾(Decorator)角色:持有一個構件(Component)對象的實例,并定義一個與抽象構件接口一致的接口。
  • 具體裝飾(ConcreteDecorator)角色:負責給構件對象“貼上”附加的責任。

示例代碼

抽象構件角色

public interface Component {
    public void sampleOpreation();
}

具體構件角色:

public class ConcreteComponent implements Component {
    @Override
    public void sampleOpreation() {
        // TODO 完成相關的業務代碼
    }
}

裝飾角色

public class Decorator implements Component {
    private Component component;
    
    public Decorator(Component component) {
        this.component = component;
    }
    
    @Override
    public void sampleOpreation() {
        //委派給構件
        component.sampleOpreation();
    }

}

具體裝飾角色

public class ConcreteDecoratorA extends Decorator {
    public ConcreteDecoratorA(Component component) {
        super(component);
    }
    
    @Override
    public void sampleOpreation() {
        super.sampleOpreation();
        //TODO 完成相關的業務代碼
    }
}
public class ConcreteDecoratorB extends Decorator {
    public ConcreteDecoratorB(Component component) {
        super(component);
    }
    
    @Override
    public void sampleOpreation() {
        super.sampleOpreation();
        //TODO 完成相關的業務代碼
    }
}

齊天大圣的例子

孫悟空有七十二般變化,他的沒一種變化都給他帶來一種附加的本領。他變成魚兒時,就已到水里游泳;他變成鳥兒時,就可以在天上飛行。

本例中,Component的角色便是由大名鼎鼎的齊天大圣扮演;ConcreteComponent的角色屬于大圣的本尊,就是猢猻本人;Decorator的角色由大圣的七十二變扮演。而ConcreteDecorator的角色就是魚兒、鳥兒等七十二般變化。

齊天大圣和七十二變的關系

示例代碼

抽象構件角色“齊天大圣”接口,定義了一個move()方法,這是所有的具體構建類和裝飾類必須實現的。

public interface TheGreatestSage {
    public void move();
}

具體構件角色“大圣本尊”,猢猻類

public class Monkey implements TheGreatestSage {
    @Override
    public void move() {
        System.out.println("Monkey move");
    }
}

抽象裝飾角色“七十二變”

public class Change implements TheGreatestSage {
    private TheGreatestSage sage;
    
    public Change(TheGreatestSage sage) {
        this.sage = sage;
    }
    @Override
    public void move() {
        this.sage.move();
    }
}

具體裝飾角色,“魚兒”和“鳥兒”

public class Fish extends Change {
    public Fish(TheGreatestSage sage) {
        super(sage);
    }
    
    @Override
    public void move() {
        super.move();
        System.out.println("Change fish move");
    }
}
public class Bird extends Change {
    public Bird(TheGreatestSage sage) {
        super(sage);
    }
    
    @Override
    public void move() {
        super.move();
        System.out.println("Change bird move");
    }
}

客戶端類

public class Client {
    public static void main(String[] args) {
        TheGreatestSage sage = new Monkey();
        //第一種寫法
        TheGreatestSage bird = new Bird(sage);
        bird.move();
        
        //第二種寫法
        TheGreatestSage fish = new Fish(new Bird(sage));
        fish.move();
    }
}

大圣本尊是ConcreteComponent類,而“鳥兒”、“魚兒”是裝飾類,要裝飾的是大圣本尊,也就是猢猻類實例。

上面的例子中,系統把大圣從一直猢猻裝飾成一只鳥兒(把鳥兒的功能加到了猢猻身上),然后又把鳥兒裝飾成一條魚兒(把魚兒的功能裝飾到猢猻+鳥兒身上),從而得到最終呈現猢猻+鳥兒+魚兒的結果。

大圣裝飾過程

如上圖所示,大圣的變化首先將鳥兒的功能附加到猢猻身上,然后又將魚兒的功能附加到猢猻+鳥兒身上。

裝飾模式的簡化

大多數情況下,裝飾模式的實現都要比上面的示意性例子要簡單。

如果只有一個ConcreteComponent類,那么可以考慮去掉抽象的Component類(接口),把Decorator作為一個ConcreteComponent的子類。如下圖所示:

合并抽象構件和具體構件

如果只有一個ConcreteDecorator類,那么就沒有必要建立一個單獨的Decorator類,而可以把DecoratorConcreteDecorator的責任合并成一個類。甚至在只有兩個ConcreteDecorator類的情況下,都可以這樣做,如下圖所示:

合并裝飾構件和具體裝飾構件

透明性的要求

裝飾模式對客戶端的透明性要求程序不要聲明給一個ConcreteComponent類型的變量,而應當聲明一個Component類型的變量。

用孫悟空的例子來說,必須永遠把孫悟空的所有變化都當成孫悟空來對待,而如果吧孫悟空變成的魚兒當成魚兒,而不是孫悟空,那么就被孫悟空騙了,而這是不應當發生的。下面的做法是對的。

TheGreatestSage sage = new Monkey();
TheGreatestSage bird = new Bird(sage);

而下面的做法是不對的:

Monkey sage = new Monkey();
Bird bird = new Bird(sage);

半透明的裝飾模式

然而,純粹的裝飾模式很難找到。裝飾模式的用意是在不改變接口的前提下,增強所考慮的類的性能。在增強性能的時候,往往要建立新的公開的方法。即便是在孫大圣的系統里,也需要新的方法。比如齊天大圣并沒有飛行的能力,而鳥兒有。這就意味著鳥兒應該有一個新的fly()方法。再比如,齊天大圣并沒有游泳的能力,而魚兒有,著就意味著魚兒應該有一個新的swim()方法。

這就導致了大多數的裝飾模式的實現都是“半透明”的,而不是完全透明的。換而言之,允許裝飾模式改變接口,增加新的方法。這意味著客戶端可以聲明ConcreteDecorator類型的變量,從而可以調用ConcreteDecorator類中才有的方法:

TheGreatestSage sage = new Monkey();
Bird bird = new Bird(sage);
bird.fly();

半透明的裝飾模式是介于裝飾模式和適配器模式之間的。適配器模式的用意是改變所考慮的類的接口,也可以通過改寫一個或幾個方法,或增加新的方法來增強或改變所考慮的類的功能。大多數的裝飾模式實際上是半透明的裝飾模式,這樣的裝飾模式也稱作半裝飾、半適配器模式。

裝飾模式的優點

  1. 裝飾模式與繼承關系的目的都是要拓展對象的功能,但是裝飾模式可以提供比繼承更多的靈活性。裝飾模式允許系統動態決定“貼上”一個需要的“裝飾”,或者“除掉”一個不需要的“裝飾”。繼承關系則不同,繼承關系是靜態的,它在系統運行前就決定了。
  2. 通過不同的具體裝飾類以及這些裝飾類的排列組合,設計師可以創造出更多不同行為的組合。

裝飾模式的缺點

由于使用裝飾模式,可以比使用繼承關系需要較少數目的類。使用較少的類,當然使設計比較易于進行。但是,在另外一方面,使用裝飾模式會產生比使用繼承關系所產生的更多的對象。而更多的對象會使得查找錯誤更為困難,特別是這些對象在看上去極為相似的時候。

裝飾設計模式在JAVA I/O庫中的應用

裝飾模式在Java語言中最著名的應用莫過于JAVA I/O標準庫的設計了。

?由于JAVA I/O庫需要很多性能的各種組合,如果這些性能都是用繼承的方法實現的,那么每一種組合都需要一種類,這樣就會造成大量性能重復的類出現。而如果采用裝飾模式,那么類的數目就會大大減少,性能的重復也可以減至最小。因此裝飾模式是JAVA I/O?庫的基本模式。

JAVA I/O庫的對象結構如下圖,由于JAVA I/O庫的對象眾多,因此只畫出InputStream的部分。

InputStream結構圖

根據上圖可以看出:

  • 抽象構建角色(Component):InputStream扮演。這是一個抽象類,為各種子類型提供統一的接口。
  • 具體構件角色(ConcreteComponent):ByteArrayInputStream、FileInputStream、StringBufferInputStream等類扮演。它們實現了抽象構件角色所規定的接口。
  • 抽象裝飾角色(Decorator):FilterInputStream、ObectInputStream等類扮演。它們實現了InputStream所規定的接口。
  • 具體裝飾角色(ConcreteDecorator):由幾個類扮演,分別是BufferedInputStream、DataInputStream以及兩個不常用到的類LineNumberInputStreamPushbackInputStream

半透明的裝飾模式

裝飾模式和適配器模式都是“包裝模式(Wrapper Pattern)”,它們都是通過封裝其他的對象達到設計的目的的,但是它們的形態有很多區別。

理想的裝飾模式在對被裝飾的對象進行功能增強的同時,要求具體構件角色、裝飾角色的接口與抽象構件角色的接口完全一致。而適配器模式則不然,一般而言,適配器模式并不要求對源對象的功能進行增強,但是會改變源對象的接口,以便和目標接口相吻合。

裝飾模式有透明和半透明兩種,這兩種的區別就在與裝飾角色的接口與抽象構件角色的接口是否完全一致。透明的裝飾模式也就時理想的裝飾模式,要求具體構件角色、裝飾角色的接口與抽象構件角色的接口完全一致。相反,如果裝飾角色的接口與抽象構件角色的接口不一致,也就說明裝飾角色的接口比抽象構件角色的接口寬的話,裝飾角色實際上已經成為了一個適配器角色,這種裝飾模式也是可以接受的,稱為“半透明”的裝飾模式,如下圖所示:

透明與半透明的裝飾模式

在適配器模式里面,適配器模式的接口通常會與目標類的接口重疊,但往往并不完全相同。換句話說,適配類的接口會比被適配的目標類接口寬。

顯然,半透明的裝飾模式實際上處于適配器模式與裝飾模式之間的灰色地帶。如果裝飾模式和適配器模式合并稱為一個“包裝模式”的話,那么半透明的裝飾模式倒是可以成為這種合并后的“包裝模式“的代表。

InputStream類型中的裝飾模式

InputStream類型中的裝飾模式是半透明的。為了說明這一點,不妨看一看裝飾模式的抽象構建角色InputStream抽象類的源代碼。這個抽象類聲明了九個方法,并給出了其中八個的實現,另外一個是抽象方法,需要子類實現。

public abstract class InputStream implements Closeable {
    public abstract int read() throws IOException;
    
    public int read(byte b[]) throws IOException {
        return read(b, 0, b.length);
    }
    
    public int read(byte b[], int off, int len) throws IOException {
        //……
    }
    
    public long skip(long n) throws IOException {
        //……
    }
    
    public int available() throws IOException {
        //……
    }
    
    public void close() throws IOException {}
    
    public synchronized void mark(int readlimit) {}
    
    public synchronized void reset() throws IOException {
        throw new IOException("mark/reset not supported");
    }
    
    public boolean markSupported() {
        return false;
    }
}

下面是作為裝飾模式的抽象裝飾角色FilterInputStream類的源代碼??梢钥闯觯?code>FilterInputStream的接口與InputStream的接口是完全一致的,也就是說,直到這一步,還是與裝飾模式相吻合的。

public class FilterInputStream extends InputStream {
    public int read() throws IOException {
        return in.read();
    }
    
    public int read(byte b[]) throws IOException {
        return read(b, 0, b.length);
    }
    
    public int read(byte b[]) throws IOException {
        return read(b, 0, b.length);
    }
    
    public long skip(long n) throws IOException {
        return in.skip(n);
    }
    
    public int available() throws IOException {
        return in.available();
    }
    
    public void close() throws IOException {
        in.close();
    }
    
    public synchronized void mark(int readlimit) {
        in.mark(readlimit);
    }
    
    public synchronized void reset() throws IOException {
        in.reset();
    }
    
    public boolean markSupported() {
        return in.markSupported();
    }
}

下面是具體裝飾角色PushbackInputStream的源代碼:

public class PushbackInputStream extends FilterInputStream {
    private void ensureOpen() throws IOException {
        if (in == null)
            throw new IOException("Stream closed");
    }
    
    public int read() throws IOException {
        //……
    }
    
    public int read(byte[] b, int off, int len) throws IOException {
        //……
    }
    
    public void unread(int b) throws IOException {
        //……
    }
    
    public void unread(byte[] b, int off, int len) throws IOException {
        //……
    }
    
    public void unread(byte[] b) throws IOException {
        unread(b, 0, b.length);
    }
    
    public int available() throws IOException {
        //……
    }
    
    public long skip(long n) throws IOException {
        //……
    }
    
    public boolean markSupported() {
        return false;
    }
    
    public synchronized void mark(int readlimit) {
    }
    
    public synchronized void reset() throws IOException {
        throw new IOException("mark/reset not supported");
    }
    
    public synchronized void close() throws IOException {
        //……
    }
}

通過查看源代碼,你會發現,這個裝飾類提供了額外的方法unread(),這就意味著PushbackInputStream是一個半透明的裝飾類。換句話說,它破壞了理想的裝飾模式的要求。如果客戶端持有一個類型為InputStream對象的引用in的話,那么如果in的真實類型是PushbackInputStream的話,只要客戶端不需要使用unread()方法,那么客戶端一般沒有問題。但是如果客戶端必須使用這個方法,就必須進行向下類型轉換。將in的類型轉換成為PushbackInputStream之后才可能調用這個方法。但是,這個類型轉換意味著客戶端必須知道它拿到的引用是指向一個類型為PushbackInputStream的對象。這就破壞了使用裝飾模式的原始用意。

現實世界與理論總歸是有一段差距的。純粹的裝飾模式在真實的系統中很難找到。一般所遇到的,都是這種半透明的裝飾模式。

示例程序

下面是使用I/O流讀取文件內容的簡單示例

import java.io.BufferedInputStream;
import java.io.DataInputStream;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;

public class InputStreamTest {
    public static void main(String[] args) {
        DataInputStream dis = null;
        try {
            dis = new DataInputStream(
                    new BufferedInputStream(
                            new FileInputStream("23.txt")
                            )
                    );
            byte[] bytes = new byte[dis.available()];
            dis.read(bytes);
            String content = new String(bytes);
            System.out.println(content);
        } catch (FileNotFoundException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            try {
                dis.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }
}

觀察上面的代碼,會發現最里層是一個FileInputStream對象,然后把它傳遞給一個BufferedInputStream對象,經過BufferedInputStream對象處理后,再將處理后的對象傳遞給DataInputStream對象進行處理。這個過程,其實就是裝飾器的組裝過程,FileInputStream對象相當于原始的被裝飾的對象,而BufferedInputStream對象和DataInputStream對象則相當于裝飾器。

參考

《JAVA與模式》之裝飾模式

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,835評論 6 534
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,676評論 3 419
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,730評論 0 380
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,118評論 1 314
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,873評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,266評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,330評論 3 443
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,482評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,036評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,846評論 3 356
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,025評論 1 371
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,575評論 5 362
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,279評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,684評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,953評論 1 289
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,751評論 3 394
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,016評論 2 375

推薦閱讀更多精彩內容

  • (轉載)原文地址 在閻宏博士的《JAVA與模式》一書中開頭是這樣描述裝飾(Decorator)模式的: 裝飾模式又...
    zjk_00閱讀 650評論 0 2
  • 本篇文章介紹一種設計模式——裝飾者模式。裝飾者模式在Java中的典型應用就是IO流,在本篇文章中將有詳細介紹。本篇...
    Ruheng閱讀 22,259評論 13 56
  • 設計模式匯總 一、基礎知識 1. 設計模式概述 定義:設計模式(Design Pattern)是一套被反復使用、多...
    MinoyJet閱讀 3,960評論 1 15
  • 1. Java基礎部分 基礎部分的順序:基本語法,類相關的語法,內部類的語法,繼承相關的語法,異常的語法,線程的語...
    子非魚_t_閱讀 31,722評論 18 399
  • reddit 上使用 Perl 6 處理日期 描述 下面的日期, 有些使用了 M D Y 格式, 有些使用了 Y...
    焉知非魚閱讀 333評論 0 1