需加装饰——装饰模式
裝飾模式指的是在不必改變?cè)?lèi)文件和使用繼承的情況下,動(dòng)態(tài)地?cái)U(kuò)展一個(gè)對(duì)象的功能。它是通過(guò)創(chuàng)建一個(gè)包裝對(duì)象,也就是裝飾來(lái)包裹真實(shí)的對(duì)象。
類(lèi)圖分析
我們先假設(shè)一個(gè)業(yè)務(wù)場(chǎng)景,有三種房子需要裝修,分別是公寓,木屋和別墅,裝修的方式有刷墻和擺滿(mǎn)鮮花。那么應(yīng)用裝飾模式以后的類(lèi)圖結(jié)構(gòu)如下所示:
這個(gè)結(jié)構(gòu)似乎與組合模式非常像,然而內(nèi)部卻大有不同。截止到Decorate部分,左上部分完全與組合模式相同,Decorate類(lèi)是裝飾類(lèi)的核心類(lèi)。
代碼展示
package pattern.decorate;/*** 裝飾類(lèi)基類(lèi),注意要繼承構(gòu)建類(lèi)基類(lèi),同時(shí)關(guān)聯(lián)一個(gè)基類(lèi)對(duì)象* * @author Evsward**/ public class Decorate extends House {protected House house;public Decorate() {// 給出一個(gè)默認(rèn)值,防止house空值異常。this.house = new Cabin();}public Decorate(House house) {this.house = house;}@Overridepublic void show() {house.show();}}這里面關(guān)鍵點(diǎn)為:
- Decorate類(lèi)也繼承了House抽象基類(lèi)。
- 同時(shí)它還包含一個(gè)House對(duì)象的成員屬性。
- 該屬性在構(gòu)造器中被初始化。
- 其復(fù)寫(xiě)的抽象方法并未實(shí)現(xiàn)任何具體內(nèi)容,而是直接調(diào)用House對(duì)象的show方法。
然后請(qǐng)看Decorate的子類(lèi)的實(shí)現(xiàn)方式,他們是如何具體的擴(kuò)展構(gòu)建類(lèi)的。
package pattern.decorate;public class GreenWallHouse extends Decorate {public GreenWallHouse(House h) {super(h);}private void painGreenOnWall() {logger.info("The wall is green now.");}@Overridepublic void show() {super.show();painGreenOnWall();}}FlowerHouse也是同理。它們的關(guān)鍵點(diǎn)是:
- 必須創(chuàng)建構(gòu)造函數(shù),將House對(duì)象傳入。
- 在實(shí)現(xiàn)抽象方法時(shí),直接調(diào)用House對(duì)象的show方法,同時(shí)加入自己的“裝飾”內(nèi)容。
角色分析
裝飾模式中重要的角色主要有兩個(gè),正是上面代碼展示部分的那兩個(gè),他們可以總結(jié)為:
優(yōu)勢(shì)
裝飾模式的架構(gòu)平淡無(wú)奇,但是卻可以不斷套用,我們來(lái)看客戶(hù)端的調(diào)用:
package pattern.decorate;import org.apache.logging.log4j.LogManager; import org.apache.logging.log4j.Logger; import org.junit.Test;public class Client {public final static Logger logger = LogManager.getLogger();@Testpublic void testDecorate() {House a = new Apartment();a.show();logger.info("---------------");House aPlus = new GreenWallHouse(a);aPlus.show();logger.info("---------------");/*** 因?yàn)樗麄兌祭^承了House,是同一個(gè)基類(lèi),所以可以無(wú)限套用裝飾類(lèi)去循環(huán)。*/House aPPlus = new GreenWallHouse(new FlowerHouse(a));aPPlus.show();} }輸出:
14:38:45[show]: This is my apartment, which is on the high floor. 14:38:45[testDecorate]: --------------- 14:38:45[show]: This is my apartment, which is on the high floor. 14:38:45[painGreenOnWall]: The wall is green now. 14:38:45[testDecorate]: --------------- 14:38:45[show]: This is my apartment, which is on the high floor. 14:38:45[decorateFlowerAround]: The room is full of flowers now. 14:38:45[painGreenOnWall]: The wall is green now.由于裝飾模式中所有新增的類(lèi)都是構(gòu)建類(lèi)的子類(lèi),并且他們每個(gè)類(lèi)都聲明了以構(gòu)建類(lèi)對(duì)象為參數(shù)的構(gòu)造函數(shù),因此,具體裝飾類(lèi)可以直接套用拓展,正如以上代碼所示。
裝飾模式,符合了面向?qū)ο笤O(shè)計(jì)原則“對(duì)修改關(guān)閉,對(duì)拓展開(kāi)放”的原則。在原有代碼完全不改動(dòng)的情況下,可以有效拓展系統(tǒng)功能。并且通過(guò)不斷的套用構(gòu)造函數(shù)的方式,使得原始構(gòu)建類(lèi)得到了多層的功能拓展,這有效地代替了多繼承。在JavaIO中,裝飾模式得到了廣泛使用。
轉(zhuǎn)載于:https://www.cnblogs.com/Evsward/p/decorate.html
總結(jié)
以上是生活随笔為你收集整理的需加装饰——装饰模式的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: maven project module
- 下一篇: Unity——用UnityEditor拷