裝飾模式

裝飾模式

一同云、概念

裝飾模式(Decorator Pattern):動(dòng)態(tài)地給一個(gè)對(duì)象增加一些額外的職責(zé),就增加對(duì)象功能來(lái)說(shuō)堵腹,裝飾模式比生成子類(lèi)實(shí)現(xiàn)更為靈活炸站。裝飾模式是一種對(duì)象結(jié)構(gòu)型模式。

定義一個(gè)抽象的裝飾類(lèi)疚顷,將具體的裝飾類(lèi)作為其子類(lèi)旱易,然后繼承具體的裝飾類(lèi)。

二腿堤、使用場(chǎng)景

  • 在不影響其他對(duì)象的情況下阀坏,以動(dòng)態(tài)、透明的方式給單個(gè)對(duì)象添加職責(zé)笆檀。
  • 當(dāng)不能采用繼承的方式對(duì)系統(tǒng)進(jìn)行擴(kuò)展或者采用繼承不利于系統(tǒng)擴(kuò)展和維護(hù)時(shí)可以使用裝飾模式忌堂。

使用裝飾模式的步驟:

  1. 在裝飾類(lèi)的內(nèi)部維護(hù)一個(gè)被裝飾類(lèi)的引用。
  2. 讓裝飾類(lèi)有一個(gè)共同的父類(lèi)或者是父接口酗洒。

三士修、UML結(jié)構(gòu)圖

裝飾模式UML.png

四、代碼示例

案例一:

Component:

public interface Component {
    public void operation();
}

Decorator:

public class Decorator implements Component{

    //維持一個(gè)對(duì)抽象構(gòu)件對(duì)象的引用
    private Component component;
    
    //注入一個(gè)抽象構(gòu)件類(lèi)型的對(duì)象(通過(guò)構(gòu)造方法或者set方法)
    public Decorator(Component component) {
        this.component = component;
    }
    
    @Override
    public void operation() {
        //調(diào)用原有業(yè)務(wù)方法樱衷,此處沒(méi)有真正的實(shí)現(xiàn)operation方法棋嘲,具體的裝飾過(guò)程交由子類(lèi)完成
        component.operation();
    }

}

ConcreateDecorator:

public class ConcreateDecorator extends Decorator{

    public ConcreateDecorator(Component component) {
        super(component);
    }
    
    public void operation(){
        //調(diào)用原有業(yè)務(wù)方法
        super.operation();
        //調(diào)用新增業(yè)務(wù)方法
        addedBehavior();
    }
    
    //新增業(yè)務(wù)方法
    public void addedBehavior(){
        
    }

}

案例二:

使用繼承的方式增強(qiáng)一個(gè)類(lèi)的功能:

package com.hcx.pattern;

import java.io.BufferedReader;
import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.IOException;
import java.io.Reader;

/**
 * 拓展BufferedReader的功能, 增強(qiáng)readLine方法箫老,返回的字符串帶有行號(hào)封字。
 * @author hcx
 *
 */
class BufferedLineNum extends BufferedReader{
    //行號(hào)
    int count = 1;

    public BufferedLineNum(Reader in) {
        super(in);
    }
    
    @Override
    public String readLine() throws IOException {
        String line = super.readLine();
        if(line==null) {
            return null;
        }
        line = count+" "+line;
        count++;
        return line;
    }

}

/**
 * 帶分號(hào)的緩沖輸入字符流
 * @author hcx
 *
 */
class BufferedSemi extends BufferedReader{

    public BufferedSemi(Reader in) {
        super(in);
    }
    
    @Override
    public String readLine() throws IOException {
        String line = super.readLine();
        if(line==null) {
            return null;
        }
        line = line+";";
        return line;
        
    }
    
}

/**
 * 帶雙引號(hào)的緩沖輸入字符流
 * @author hcx
 *
 */
class BufferedQuto extends BufferedReader{

    public BufferedQuto(Reader in) {
        super(in);
    }
    
    @Override
    public String readLine() throws IOException {
        String line = super.readLine();
        if(line == null) {
            return null;
        }
        line = "\""+line+"\"";
        return line;
    }
    
}
public class Demo1 {
    
    public static void main(String[] args) throws IOException {
        File file = new File("F:\\Demo1.java");
        //建立數(shù)據(jù)的輸入通道
        FileReader fileReader = new FileReader(file);
        //建立帶行號(hào)的緩沖輸入字符流
        BufferedLineNum bufferedLineNum = new BufferedLineNum(fileReader);
        
        //帶有分號(hào)的緩沖輸入字符流
        BufferedSemi bufferedSemi = new BufferedSemi(fileReader);
        
        //帶有雙引號(hào)的緩沖輸入字符流
        BufferedQuto bufferedQuto = new BufferedQuto(fileReader);
        
        String line = null;
        while((line = bufferedLineNum.readLine())!=null) {
            System.out.println(line);
        }
    }

}

使用裝飾模式增強(qiáng)一個(gè)類(lèi)的功能:

package com.hcx.pattern;

import java.io.BufferedReader;
import java.io.File;
import java.io.FileReader;
import java.io.IOException;
import java.io.Reader;

/**
 * 帶行號(hào)的緩沖輸入字符流
 * @author hcx
 *
 */
class BufferedLineNum2 extends BufferedReader{
    
    //在內(nèi)部維護(hù)一個(gè)被裝飾類(lèi)的引用。
    BufferedReader bufferedReader;
    
    int count = 1;

    public BufferedLineNum2(BufferedReader bufferedReader) {
        // 注意: 該語(yǔ)句沒(méi)有任何的作用,只不過(guò)是為了讓代碼不報(bào)錯(cuò)阔籽。
        super(bufferedReader);
        this.bufferedReader = bufferedReader;
    }
    
    public String readLine() throws IOException {
        String line = bufferedReader.readLine();
        if(line==null) {
            return null;
        }
        line = count+" "+line;
        count++;
        return line;
        
    }
    
}

/**
 * 帶分號(hào)緩沖輸入字符流
 * 繼承的原因:為了讓這些裝飾類(lèi)的對(duì)象可以作為參數(shù)進(jìn)行傳遞流妻,達(dá)到互相裝飾的效果。
 * @author hcx
 *
 */
class BufferedSemi2 extends BufferedReader{
    
    //在內(nèi)部維護(hù)一個(gè)被裝飾類(lèi)的引用笆制。
    BufferedReader bufferedReader;
    
    public BufferedSemi2(BufferedReader bufferedReader) {
        //BufferReader沒(méi)有無(wú)參的構(gòu)造方法绅这,繼承了BufferedReader,所以要指定調(diào)用父類(lèi)的帶參的構(gòu)造方法
        super(bufferedReader);// 注意: 該語(yǔ)句沒(méi)有任何的作用在辆,只不過(guò)是為了讓代碼不報(bào)錯(cuò)证薇。
        this.bufferedReader = bufferedReader;
    }
    
    public String readLine() throws IOException{
        //創(chuàng)建該類(lèi)對(duì)象時(shí),如果傳入的是buffereLineNum匆篓,則這里的ReadLine方法是調(diào)用了buffereLineNum的readLine方法.
        String line = bufferedReader.readLine();  
        if(line==null){
            return null;
        }
        line = line +";";
        return line;
    }
}

/**
 * 帶雙引號(hào)緩沖輸入字符流
 * @author hcx
 *
 */
class BufferedQuto2 extends BufferedReader{
    
    //在內(nèi)部維護(hù)一個(gè)被裝飾的類(lèi)
    BufferedReader bufferedReader;
    
    public BufferedQuto2(BufferedReader bufferedReader){  //new  BufferedSemi2();
        super(bufferedReader) ; //只是為了讓代碼不報(bào)錯(cuò)
        this.bufferedReader = bufferedReader;
    }
    
    public String readLine() throws IOException{
        //創(chuàng)建該類(lèi)對(duì)象時(shí)浑度,如果傳入的是bufferedSemi2,則這里的ReadLine方法是調(diào)用了bufferedSemi2的readLine方法.
        String line = bufferedReader.readLine();  
        if(line==null){
            return null;
        }
        line = "\""+line +"\"";
        return line;
    }
    
    
}


public class Demo2 {
    
    public static void main(String[] args) throws IOException {
        File file = new File("F:\\Demo1.java");
        FileReader fileReader = new FileReader(file);
        //建立緩沖輸入字符流
        BufferedReader bufferedReader = new BufferedReader(fileReader);
        //建立帶行號(hào)的緩沖輸入字符流
        BufferedLineNum2 bufferedLineNum = new BufferedLineNum2(bufferedReader);
        
        //帶分號(hào)的緩沖輸入字符流
        BufferedSemi2 bufferedSemi2 = new BufferedSemi2(bufferedLineNum);
        
        //帶雙引號(hào)的緩沖輸入字符流
        BufferedQuto2 bufferedQuto2 = new  BufferedQuto2(bufferedSemi2);
        
        String line = null;
        while((line = bufferedQuto2.readLine())!=null){
            System.out.println(line);
        }
    }

}

案例三:

一家三口每個(gè)人都會(huì)工作鸦概,兒子的工作就是畫(huà)畫(huà)箩张,母親的工作就是在兒子的基礎(chǔ)上做一個(gè)增強(qiáng),不單止可以畫(huà)畫(huà)窗市,還可以上涂料先慷。爸爸的工作就是在媽媽基礎(chǔ)上做了增強(qiáng),就是上畫(huà)框咨察。

interface Work{
    
    public void work();
}

class Son implements Work{

    @Override
    public void work() {
        System.out.println("畫(huà)畫(huà)");
    }
}


class Mather implements Work{

    //需要被增強(qiáng)的類(lèi)论熙。
    Work worker;
    
    public Mather(Work worker){
        this.worker = worker;
    }
    
    @Override
    public void work() {
        worker.work();
        System.out.println("給畫(huà)上顏色");
    }
}


class Father implements Work{

    //需要被增強(qiáng)的類(lèi)的引用
    Work worker;
    
    public Father(Work worker){
        this.worker = worker;
    }
    
    
    @Override
    public void work() {
        worker.work();
        System.out.println("上畫(huà)框");
    }
    
}

public class Demo {
    
    public static void main(String[] args) {
        Son s = new Son();
//      s.work();
        Mather m = new Mather(s);
//      m.work();
        Father f = new Father(s);
        f.work();
        
    }
}

總結(jié):

繼承實(shí)現(xiàn)的增強(qiáng)類(lèi)和裝飾模式實(shí)現(xiàn)的增強(qiáng)類(lèi)有何區(qū)別?

繼承實(shí)現(xiàn)的增強(qiáng)類(lèi):

  • 優(yōu)點(diǎn):代碼結(jié)構(gòu)清晰摄狱,而且實(shí)現(xiàn)簡(jiǎn)單.
  • 缺點(diǎn):對(duì)于每一個(gè)的需要增強(qiáng)的類(lèi)都要?jiǎng)?chuàng)建具體的子類(lèi)來(lái)幫助其增強(qiáng)脓诡,這樣會(huì)導(dǎo)致繼承體系過(guò)于龐大。

裝飾模式實(shí)現(xiàn)的增強(qiáng)類(lèi):

  • 優(yōu)點(diǎn):內(nèi)部可以通過(guò)多態(tài)技術(shù)對(duì)多個(gè)需要增強(qiáng)的類(lèi)進(jìn)行增強(qiáng)媒役,可以使這些裝飾類(lèi)達(dá)到互相裝飾的效果誉券。使用比較靈活。
  • 缺點(diǎn):需要內(nèi)部通過(guò)多態(tài)技術(shù)維護(hù)需要被增強(qiáng)的類(lèi)的實(shí)例刊愚。進(jìn)而使得代碼稍微復(fù)雜踊跟。

五、裝飾模式的優(yōu)點(diǎn)

  • 對(duì)于擴(kuò)展一個(gè)對(duì)象的功能鸥诽,裝飾模式比繼承更加靈活商玫,不會(huì)導(dǎo)致類(lèi)的個(gè)數(shù)急劇增長(zhǎng)。
  • 可以通過(guò)一種動(dòng)態(tài)的方式來(lái)擴(kuò)展一個(gè)對(duì)象的功能
  • 可以對(duì)一個(gè)對(duì)象進(jìn)行多次裝飾牡借,通過(guò)使用不同的具體裝飾類(lèi)以及這些裝飾類(lèi)的排列組合拳昌。
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市钠龙,隨后出現(xiàn)的幾起案子炬藤,更是在濱河造成了極大的恐慌御铃,老刑警劉巖,帶你破解...
    沈念sama閱讀 211,348評(píng)論 6 491
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件沈矿,死亡現(xiàn)場(chǎng)離奇詭異上真,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)羹膳,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,122評(píng)論 2 385
  • 文/潘曉璐 我一進(jìn)店門(mén)睡互,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人陵像,你說(shuō)我怎么就攤上這事就珠。” “怎么了醒颖?”我有些...
    開(kāi)封第一講書(shū)人閱讀 156,936評(píng)論 0 347
  • 文/不壞的土叔 我叫張陵妻怎,是天一觀(guān)的道長(zhǎng)。 經(jīng)常有香客問(wèn)我泞歉,道長(zhǎng)蹂季,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 56,427評(píng)論 1 283
  • 正文 為了忘掉前任疏日,我火速辦了婚禮,結(jié)果婚禮上撒汉,老公的妹妹穿的比我還像新娘沟优。我一直安慰自己,他們只是感情好睬辐,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,467評(píng)論 6 385
  • 文/花漫 我一把揭開(kāi)白布挠阁。 她就那樣靜靜地躺著,像睡著了一般溯饵。 火紅的嫁衣襯著肌膚如雪侵俗。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 49,785評(píng)論 1 290
  • 那天丰刊,我揣著相機(jī)與錄音隘谣,去河邊找鬼。 笑死啄巧,一個(gè)胖子當(dāng)著我的面吹牛寻歧,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播秩仆,決...
    沈念sama閱讀 38,931評(píng)論 3 406
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼码泛,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了澄耍?” 一聲冷哼從身側(cè)響起噪珊,我...
    開(kāi)封第一講書(shū)人閱讀 37,696評(píng)論 0 266
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤晌缘,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后痢站,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體磷箕,經(jīng)...
    沈念sama閱讀 44,141評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,483評(píng)論 2 327
  • 正文 我和宋清朗相戀三年瑟押,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了搀捷。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,625評(píng)論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡多望,死狀恐怖嫩舟,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情怀偷,我是刑警寧澤家厌,帶...
    沈念sama閱讀 34,291評(píng)論 4 329
  • 正文 年R本政府宣布,位于F島的核電站椎工,受9級(jí)特大地震影響饭于,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜维蒙,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,892評(píng)論 3 312
  • 文/蒙蒙 一掰吕、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧颅痊,春花似錦殖熟、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,741評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至舰罚,卻和暖如春纽门,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背营罢。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 31,977評(píng)論 1 265
  • 我被黑心中介騙來(lái)泰國(guó)打工赏陵, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人饲漾。 一個(gè)月前我還...
    沈念sama閱讀 46,324評(píng)論 2 360
  • 正文 我出身青樓瘟滨,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親能颁。 傳聞我的和親對(duì)象是個(gè)殘疾皇子杂瘸,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,492評(píng)論 2 348

推薦閱讀更多精彩內(nèi)容