Java設計模式(3 / 23):裝飾者模式

文章目錄

    • 定義
    • 案例1:三點幾啦
      • 首次嘗試
      • 再次嘗試
      • 設計原則:類應該對擴展開放,對修改關閉
      • 嘗用裝飾者模式
      • 裝飾者模式特征
      • 本例的類圖
      • 放碼過來
        • 飲料類
          • HouseBlend
          • DarkRoast
          • Espresso
          • Decaf
          • 調料裝飾類
            • Milk
            • Mocha
            • Soy
            • Whip
        • 運行測試類
    • 案例2:編寫自己的Java I/0裝飾者
      • JDK中的IO流概述
        • 放碼過來
          • LowerCaseInputStream
          • 創建一測試文件D:\\test.txt
          • 運行測試類
    • 參考資料

定義

裝飾者(Decorator)模式動態地將責任附加到對象上。若要擴展功能,裝飾者提供了比繼承更有彈性的替代方案

案例1:三點幾啦

更新咖啡連鎖店的訂單系統,原先類的設計:

咖啡店為拓展業務,允許顧客在飲料上添加各種調料,如:

  1. 蒸奶 Steamed Milk
  2. 豆漿 Soy
  3. 摩卡(巧克力風味) Mocha
  4. 覆蓋奶泡

加入的調料收取不同的費用。

首次嘗試

類數量爆炸

這違背嚴重兩條設計原則

  1. 多用組合,少用繼承。
  2. 為了交互對象之間的松耦合設計而努力。

再次嘗試

利用實例變量和繼承,追蹤這些調料:

這次嘗試的局限性:

  1. 調料價錢的改變會使我們更改現有代碼。
  2. 一旦出現新的調料,我們就需要加上新的方法,并改變超類中的cost()方法。
  3. 以后可能會開發出新飲料。對這些飲料而言(例如:冰茶),某些調料可能并不適合,但是在這個設計方式中,Tea(茶)子類仍將繼承那些不適合的方法,例如:hasWhip()(加奶泡)。
  4. 萬一顧客想要雙倍摩卡咖啡,怎么辦?

設計原則:類應該對擴展開放,對修改關閉

我們的目標是允許類容易擴展,在不修改現有代碼的情況下,就可搭配新的行為。

如能實現這樣的目標,這樣的設計具有彈性可以應對改變,可以接受新的功能來應對改變的需求。

雖然似乎有點矛盾,但是的確有一些技術可以允許在不直接修改代碼的情況下對其進行擴展。

在選擇需要被擴展的代碼部分時要小心。每個地方都采用開放-關閉原則,是一種浪費,也沒必要,還會導致代碼變得復雜且難以理解。

(MyNote:隨機應變)

嘗用裝飾者模式

我們要以飲料為主體,然后在運行時以調料來“裝飾”(decorate)飲料。比方說,如果顧客想要摩卡
和奶泡深焙咖啡,那么,要做的是:

1.拿一個深焙咖啡(DarkRoast)對象

2.以摩卡(Mocha)對象裝飾它

3.以奶泡(Whip)對象裝飾它

4.調用cost()方法,并依賴委托(delegate)將調料的價
錢加上去。

裝飾者模式特征

  • 裝飾者和被裝飾對象有相同的超類型。

  • 你可以用一個或多個裝飾者包裝一個對象。

  • 既然裝飾者和被裝飾對象有相同的超類型,所以在任何需要原始對象(被包裝的)的場合,可以用裝飾過的對象代替它。

  • 裝飾者可以在所委托被裝飾者的行為之前與/或之后,加上自己的行為,以達到特定的目的。

  • 對象可以在任何時候被裝飾,所以可以在運行時動態地、不限量地用你喜歡的裝飾者來裝飾對象。

本例的類圖

放碼過來

飲料類

public abstract class Beverage {String description = "Unknown Beverage";public String getDescription() {return description;}public abstract double cost();
}
HouseBlend
public class HouseBlend extends Beverage {public HouseBlend() {description = "House Blend Coffee";}public double cost() {return .89;}
}
DarkRoast
public class DarkRoast extends Beverage {public DarkRoast() {description = "Dark Roast Coffee";}public double cost() {return .99;}
}
Espresso
public class Espresso extends Beverage {public Espresso() {description = "Espresso";}public double cost() {return 1.99;}
}
Decaf
public class Decaf extends Beverage {public Decaf() {description = "Decaf Coffee";}public double cost() {return 1.05;}
}
調料裝飾類
public abstract class CondimentDecorator extends Beverage {Beverage beverage;public abstract String getDescription();
}
Milk
public class Milk extends CondimentDecorator {public Milk(Beverage beverage) {this.beverage = beverage;}public String getDescription() {return beverage.getDescription() + ", Milk";}public double cost() {return .10 + beverage.cost();}
}
Mocha
public class Mocha extends CondimentDecorator {public Mocha(Beverage beverage) {this.beverage = beverage;}public String getDescription() {return beverage.getDescription() + ", Mocha";}public double cost() {return .20 + beverage.cost();}
}
Soy
public class Soy extends CondimentDecorator {public Soy(Beverage beverage) {this.beverage = beverage;}public String getDescription() {return beverage.getDescription() + ", Soy";}public double cost() {return .15 + beverage.cost();}
}
Whip
public class Whip extends CondimentDecorator {public Whip(Beverage beverage) {this.beverage = beverage;}public String getDescription() {return beverage.getDescription() + ", Whip";}public double cost() {return .10 + beverage.cost();}
}

運行測試類

public class StarbuzzCoffee {public static void main(String args[]) {Beverage beverage = new Espresso();System.out.println(beverage.getDescription() + " $" + beverage.cost());System.out.println("---");Beverage beverage2 = new DarkRoast();beverage2 = new Mocha(beverage2);beverage2 = new Mocha(beverage2);beverage2 = new Whip(beverage2);System.out.println(beverage2.getDescription() + " $" + beverage2.cost());System.out.println("---");Beverage beverage3 = new HouseBlend();beverage3 = new Soy(beverage3);beverage3 = new Mocha(beverage3);beverage3 = new Whip(beverage3);System.out.println(beverage3.getDescription() + " $" + beverage3.cost());}
}

運行結果:

Espresso $1.99
---
Dark Roast Coffee, Mocha, Mocha, Whip $1.49
---
House Blend Coffee, Soy, Mocha, Whip $1.34

案例2:編寫自己的Java I/0裝飾者

JDK中的IO流概述

java.io包內的類中許多類都是裝飾者。

下面是一個典型的對象集合,用裝飾者來將功能結合起來,
以讀取文件數據:

BufferedInputStream及LineNumberInputStream都擴展自FilterInputStream,而FilterInputStream是一個抽象的裝飾類。

java.io其實沒有多大的差異。我們把java.io API范圍縮小,讓你容易查看它的文件,并組合各種“輸入”流裝飾者來符合你的用途。

你會發現輸出流(OutputStream)的設計方式也是一樣的。你可能還會發現Reader/Writer流(作為基于字符數據的輸入輸出)和輸入流/輸出流的類相當類似(雖然有一些小差異和不一致之處,但是相當雷同,所以你應該可以了解這些類)。

但是Java I/O也引出裝飾者模式的一個缺點:利用裝飾者模式,常常造成設計中有大量的小類,數量實在太多,可能會造成使用此API程序員的困擾。

但是,現在你已經了解了裝飾者的工作原理,以后當使用別人的大量裝飾的API時,就可以很容易地辨別出他們的裝飾者類是如何組織的,以方便用包裝方式取得想要的行為。

放碼過來

這個想法怎么樣:編寫一個裝飾者,把輸入流內的所有大寫字符轉成小寫。

舉例:當讀取“I know the Decorator Pattern thereforeI RULE!”,裝飾者會將它轉成“i know thedecorator pattern therefore i rule ! ”

LowerCaseInputStream

擴展FilterInputStream,這是所有InputStream的抽象裝飾者:

import java.io.*;public class LowerCaseInputStream extends FilterInputStream {public LowerCaseInputStream(InputStream in) {super(in);}public int read() throws IOException {int c = in.read();return (c == -1 ? c : Character.toLowerCase((char)c));}public int read(byte[] b, int offset, int len) throws IOException {int result = in.read(b, offset, len);for (int i = offset; i < offset+result; i++) {b[i] = (byte)Character.toLowerCase((char)b[i]);}return result;}
}
創建一測試文件D:\test.txt
I know the Decorator Pattern therefore I RULE!
運行測試類
import java.io.*;public class InputTest {public static void main(String[] args) throws IOException {int c;InputStream in = null;try {in = new LowerCaseInputStream( new BufferedInputStream(new FileInputStream("D:\\text.txt")));while((c = in.read()) >= 0) {System.out.print((char)c);}} catch (IOException e) {e.printStackTrace();} finally {if (in != null) { in.close(); }}System.out.println();System.out.println("---");//try (InputStream in2 = new LowerCaseInputStream(new BufferedInputStream(new FileInputStream("D:\\text.txt")))) {while((c = in2.read()) >= 0) {System.out.print((char)c);}} catch (IOException e) {e.printStackTrace();}}
}

運行結果:

i know the decorator pattern therefore i rule!
---
i know the decorator pattern therefore i rule!

參考資料

  1. 《Head First 設計模式》

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/445597.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/445597.shtml
英文地址,請注明出處:http://en.pswp.cn/news/445597.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

c語言知識體系

原文&#xff1a;https://blog.csdn.net/lf_2016/article/details/80126296#comments

《游戲編程入門 4th》筆記(1 / 14):Windows初步

文章目錄Windows編程概述獲取Windows理解Windows消息機制多任務多線程事件處理DirectX快速概覽Direct3D是什么Window程序基礎創建第一個Win32項目理解WinMainWinMain函數調用完整的WinMainGetMessage函數調用尋求幫助Windows編程概述 DirectX&#xff0c;流行的游戲編程庫。它…

17校招真題題集(1)1-5

注&#xff1a;本系列題目全是按照通過率降序來排列的&#xff0c;基本保證題目難度遞增。 1、 題目名稱&#xff1a;游戲任務標記 來源&#xff1a;騰訊 題目描述 游戲里面有很多各式各樣的任務&#xff0c;其中有一種任務玩家只能做一次&#xff0c;這類任務一共有1024個…

《游戲編程入門 4th》筆記(2 / 14):監聽Windows消息

文章目錄編寫一個Windows程序理解InitInstanceInitInstance函數調用InitInstance的結構理解MyRegisterClassMyRegisterClass函數調用MyRegisterClass的作用揭露WinProc的秘密WinProc函數調用WinProc的大秘密什么是游戲循環The Old WinMain對持續性的需要實時終止器WinMain和循環…

17校招真題題集(2)6-10

注&#xff1a;本系列題目全是按照通過率降序來排列的&#xff0c;基本保證題目難度遞增。 6、 題目名稱&#xff1a;Fibonacci數列 來源&#xff1a;網易 題目描述 Fibonacci數列是這樣定義的&#xff1a; F[0] 0 F[1] 1 for each i ≥ 2: F[i] F[i-1] F[i-2] 因此&am…

QT5的數據庫

#include <QtSql> QT sql QSqlDatabase類實現了數據庫連接的操作 QSqlQuery類執行SQL語句 QSqlRecord類封裝數據庫所有記錄 QSqlDatabase類 [cpp] view plaincopy print?QSqlDatabase db QSqlDatabase::addDatabase("QOCI"); db.setHostName("localh…

數據結構課上筆記6

本節課介紹了單鏈表的操作實現細節&#xff0c;介紹了靜態鏈表。 鏈表帶頭的作用&#xff1a;對鏈表進行操作時&#xff0c;可以對空表、非空表的情況以及 對首元結點進行統一處理&#xff0c;編程更方便。 下面給出帶頭的單鏈表實現思路&#xff1a; 按下標查找&#xff1a; …

《Unity2018入門與實戰》筆記(9 / 9):個人總結

個人總結 腳本語言學習的竅門 盡可能多讀、多寫、多說腳本語言&#xff01; Link 游戲制作步驟 設計游戲時一般會遵循5個步驟&#xff1a; 羅列出畫面上所有的對象。確定游戲對象運行需要哪些控制器腳本。確定自動生成游戲對象需要哪些生成器腳本。準備好用于更新UI的調度…

17校招真題題集(3)11-15

注&#xff1a;本系列題目全是按照通過率降序來排列的&#xff0c;基本保證題目難度遞增。 11、 題目名稱&#xff1a;買蘋果 來源&#xff1a;網易 題目描述 小易去附近的商店買蘋果&#xff0c;奸詐的商販使用了捆綁交易&#xff0c;只提供6個每袋和8個每袋的包裝(包裝不…

Qt學習:QDomDocument

QDomDocument類代表了一個XML文件 QDomDocument類代表整個的XML文件。概念上講&#xff1a;它是文檔樹的根節點&#xff0c;并提供了文檔數據的基本訪問方法。由于元素、文本節點、注釋、指令執行等等不可能脫離一個文檔的上下文&#xff0c;所以文檔類也包含了需要用來創建這些…

《事實:用數據思考,避免情緒化決策》筆記

文章目錄一分為二負面思維直線思維恐懼本能規模錯覺以偏概全命中注定單一視角歸咎他人情急生亂一分為二 要做到實事求是&#xff0c; 就要做到當你聽到一分為二的說法時&#xff0c; 你就能迅速認識到這種說法描述的是一種兩極分化的圖畫&#xff0c; 而兩極之間存在一道巨大的…

順序存儲線性表實現

在計算機中用一組地址連續的存儲單元依次存儲線性表的各個數據元素,稱作線性表的順序存儲結構。 順序存儲結構的主要優點是節省存儲空間&#xff0c;因為分配給數據的存儲單元全用存放結點的數據&#xff08;不考慮c/c語言中數組需指定大小的情況&#xff09;&#xff0c;結點之…

QT5生成.exe文件時,出現缺少QT5core.dll文件解決方法

在 http://qt-project.org/downloads 下載Qt SDK安裝需要Qt版本。在QtCreator下&#xff0c;程序可以正常運行&#xff0c;但是當關閉QtCreator后&#xff0c;在DeBug目錄下再運行相應的*.exe程序時&#xff0c;會提示缺少Qt5Core.dll錯誤。解決方法&#xff1a;添加電腦環境變…

《基于Java實現的遺傳算法》筆記(7 / 7):個人總結

文章目錄為何采用遺傳算法哪些問題適合用遺傳算法解決遺傳算法基本術語一般遺傳算法的過程基本遺傳算法的偽代碼為何采用遺傳算法 遺傳算法是機器學習的子集。在實踐中&#xff0c;遺傳算法通常不是用來解決單一的、特定問題的最好算法。對任何一個問題&#xff0c;幾乎總有更…

單鏈表不帶頭標準c語言實現

鏈表是一種物理存儲單元上非連續、非順序的存儲結構&#xff0c;數據元素的邏輯順序是通過鏈表中的指針鏈接次序實現的。鏈表由一系列結點&#xff08;鏈表中每一個元素稱為結點&#xff09;組成&#xff0c;結點可以在運行時動態生成。每個結點包括兩個部分&#xff1a;一個是…

Java設計模式(4 / 23):單例模式

文章目錄單例模式的應用場景餓漢式單例模式懶漢式單例模式改進&#xff1a;synchronized改進&#xff1a;雙重檢查鎖改進&#xff1a;靜態內部類破壞單例用反射破壞單例用序列化破壞單例解密注冊式單例模式枚舉式單例模式解密容器式單例線程單例實現ThreadLocal單例模式小結參考…

約瑟夫環-(數組、循環鏈表、數學)

約瑟夫環&#xff08;約瑟夫問題&#xff09;是一個數學的應用問題&#xff1a;已知n個人&#xff08;以編號1&#xff0c;2&#xff0c;3...n分別表示&#xff09;圍坐在一張圓桌周圍。從編號為k的人開始報數&#xff0c;數到m的那個人出列&#xff1b;他的下一個人又從1開始報…

Ubuntu麒麟下搭建FTP服務

一.怎么搭建FTP服務&#xff1a; 第一步>>更新庫 linuxidclinuxidc:~$ sudo apt-get update 第二步>>采用如下命令安裝VSFTPD的包 linuxidclinuxidc:~$ sudo apt-get install vsftpd 第三步>>安裝完成后打開 /etc/vsftpd.conf 文件&#xff0c;按如下所述…

《數據結構上機實驗(C語言實現)》筆記(1 / 12):緒論

文章目錄驗證性實驗求1~n的連續整數和說明放碼結果常見算法時間函數的增長趨勢分析說明放碼結果設計性實驗求素數個數說明放碼結果求連續整數階乘的和說明放碼結果驗證性實驗 求1~n的連續整數和 說明 對于給定的正整數n&#xff0c;求12…n12…n12…n&#xff0c;采用逐個累…

線性表實現一元多項式操作

數組存放&#xff1a; 不需要記錄冪&#xff0c;下標就是。 比如1&#xff0c;2&#xff0c;3&#xff0c;5表示12x3x^25x^3 有了思路&#xff0c;我們很容易定義結構 typedef struct node{float * coef;//系數數組int maxSize;//最大容量int order;//最高階數 }Polynomial…