什么是裝飾器模式
裝飾器模式的概念
從概念上講郑趁,是在實(shí)現(xiàn)一個(gè)抽象內(nèi)容的同時(shí)纬凤,以增強(qiáng)或擴(kuò)展其功能為目的,對(duì)它進(jìn)行一層層笤昨,可互相嵌套式的祖驱,能靈活添加或去除且順序可變的包裝。
裝飾器的UML結(jié)構(gòu)圖
從UML類圖結(jié)構(gòu)上來(lái)說(shuō)瞒窒,它是類的多態(tài)特性的擴(kuò)展捺僻。除了‘具體構(gòu)建對(duì)象’對(duì)‘抽象構(gòu)建接口’的獨(dú)立實(shí)現(xiàn)外,在‘裝飾器(接口實(shí)現(xiàn)類)’構(gòu)建時(shí)還允許將‘抽象構(gòu)建接口’作為構(gòu)建方法的入?yún)⒊绮茫弧b飾器’當(dāng)作成員屬性持有匕坯,最終由‘實(shí)際的裝飾器對(duì)象’繼承‘裝飾器’來(lái)實(shí)現(xiàn)具體的功能,并允許‘實(shí)際的裝飾器對(duì)象’可以相互嵌套已實(shí)現(xiàn)功能的疊加和擴(kuò)展拔稳。
圖例說(shuō)明:
Component:抽象構(gòu)建接口葛峻。描述需要抽象的內(nèi)容
ConcreteComponent:獨(dú)立的構(gòu)建對(duì)象。通過(guò)Java泛化來(lái)實(shí)現(xiàn)抽象構(gòu)建接口巴比。
Decorator:裝飾器抽象父類术奖。所有的子裝飾器對(duì)象的統(tǒng)一父類礁遵,需要內(nèi)部持有一個(gè)Component并在構(gòu)建方法中入?yún)⑦@個(gè)Component,并同時(shí)實(shí)現(xiàn)Component接口采记。
ConreteDecoratorA佣耐、ConreteDecoratorB:實(shí)際的裝飾器對(duì)象。具體負(fù)責(zé)擴(kuò)展Component的功能挺庞。
Java中如何使用裝飾器模式
舉一個(gè)關(guān)于Drink的例子說(shuō)明
日常生活中的飲品種類多不勝數(shù)晰赞,但是它們的配料內(nèi)容大部分是相同的,比如綠茶就是開(kāi)水沖茶葉选侨,奶茶就是已經(jīng)制作好的綠茶中添加鮮牛奶掖鱼,當(dāng)然奶茶也可以先煮鮮牛奶,再放入茶葉援制。像生活中這類制作順序可靈活變化的事物戏挡,就可以用裝飾器模式來(lái)描述。
裝飾器模式中都有一個(gè)同源的接口類(抽象構(gòu)建接口類)晨仑,描述需要構(gòu)想的內(nèi)容褐墅。在這里用IDrink這個(gè)接口來(lái)描述‘飲品’這一抽象類。
裝飾器模式中還需要一個(gè)統(tǒng)一的裝飾器抽象父類洪己,這里用DrinkDecorator抽象類來(lái)代表:
例子中WaterDrink是直接用繼承的方式實(shí)現(xiàn)IDrink妥凳,這里作為最地基礎(chǔ)的原料(飲品當(dāng)然水是最基本的嘛)。
之后的TeaDrink(茶飲)答捕、MilkyDrink(奶飲)逝钥、FruitDrink(果飲)都是用WaterDrink來(lái)制作的(從UML上來(lái)說(shuō),就是內(nèi)部持有IDrink的實(shí)現(xiàn)類)
就像例子開(kāi)頭舉得奶茶的例子拱镐,就可以用UML圖做如下表達(dá):
代碼實(shí)現(xiàn)如下:
執(zhí)行結(jié)果:
成分配料:飲用水;高品質(zhì)茶葉;鮮牛奶;
制作方法:->添加飲用水->開(kāi)水煮茶葉->混合新鮮牛奶
同理艘款,如果想更復(fù)雜一些,比如水果味奶茶沃琅,可以用UML圖描述如下:
代碼如下哗咆,結(jié)果就不重復(fù)了
Java中的裝飾器模式的優(yōu)缺點(diǎn)
靈活的組合是最大的優(yōu)點(diǎn)。裝飾器模式與類的泛化都是對(duì)類功能的擴(kuò)展益眉,但使用裝飾器模式晌柬,從代碼維護(hù)上來(lái)講,可以更簡(jiǎn)單的為對(duì)象“添加一個(gè)裝飾(包裹一層功能外衣)”郭脂,“去除一個(gè)裝飾(脫下一層)”空繁,順序也可以方便的調(diào)整,可以通過(guò)不同的組合實(shí)現(xiàn)各色各樣的業(yè)務(wù)場(chǎng)景朱庆。
缺點(diǎn)也很明顯,它會(huì)產(chǎn)生很多的裝飾對(duì)象闷祥。如果不加以設(shè)計(jì)和充分描述娱颊,其他程序員會(huì)較難理解并使用各個(gè)裝飾對(duì)象傲诵。