Java裝飾模式

来源:https://www.cnblogs.com/dongguangming/archive/2020/05/17/12901589.html
-Advertisement-
Play Games

你在山上看風景,看風景的人在山上看你。明月裝飾了你的窗子,你裝飾了別人的夢。 裝飾器模式(Decorator Pattern),別名又叫包裝者模式(wapper),允許向一個現有的對象添加新的功能,同時又不改變其結構。這種類型的設計模式屬於結構型模式,它是作為現有的類的一個包裝,不同於代理。 這種模 ...


你在山上看風景,看風景的人在山上看你。明月裝飾了你的窗子,你裝飾了別人的夢。

 

裝飾器模式(Decorator Pattern),別名又叫包裝者模式(wapper),允許向一個現有的對象添加新的功能,同時又不改變其結構。這種類型的設計模式屬於結構型模式,它是作為現有的類的一個包裝,不同於代理。

這種模式創建了一個裝飾類,用來包裝原有的類,併在保持類方法簽名完整性的前提下,提供了額外的功能。

介紹

意圖:動態地給一個對象添加一些額外的職責。就增加功能來說,裝飾器模式相比生成子類更為靈活。

主要解決:一般我們為了擴展一個類經常使用繼承方式實現,由於繼承為類引入靜態特征,並且隨著擴展功能的增多,子類會很膨脹。

何時使用:在不想增加很多子類的情況下擴展類。

如何解決:將具體功能職責劃分,同時繼承裝飾者模式。

關鍵代碼: 1、Component 類充當抽象角色,不應該具體實現。 2、修飾類引用和繼承 Component 類,具體擴展類重寫父類方法。

優點:裝飾類和被裝飾類可以獨立發展,不會相互耦合,裝飾模式是繼承的一個替代模式,裝飾模式可以動態擴展一個實現類的功能。

缺點:多層裝飾比較複雜。

使用場景: 1、擴展一個類的功能。 2、動態增加功能,動態撤銷。

註意事項:可代替繼承。

結構圖

參與者:

Component:定義一個對象介面,可以給這些對象動態的添加職責

ConcreteComponent: 定義一個對象,可以給這個對象添加一些職責

Decorator:維持一個指向Component對象的指針,並定義一個與Component介面一致的介面

ConcreteDecoratorA,ConcreteDecoratorB。。。CD等:向組件添加職責

 

我們通過下麵的實例來演示裝飾器模式的用法,給圖像修飾比如上色、裝框、然後列印。

1. 定義一個抽象類:

/**
 * @author dgm
 * @describe "圖片圖像,同比Conpenent"
 */
public abstract class PhotoImage {
	//圖像描述
	public abstract String getDescription();
}

2.  創建抽象類的包裝實現類。

/**
 * 
 * @author dgm
 * @describe "類比ConcreteComponent"
 */
public class PhotoImageWrapper extends PhotoImage {

	// 圖像標題
	String title;
	// 圖片文件名
	String fileName;
	// 圖像寬高
	int pixWidth, pixHeight;

	public PhotoImageWrapper() {
		// Empty; used in Decorators.
	}

	public PhotoImageWrapper(String title, String fileName) {
		super();
		this.title = title;
		this.fileName = fileName;
	}

	@Override
	public String toString() {
		return getDescription();
	}

	public String getTitle() {
		return title;
	}

	public void setTitle(String title) {
		this.title = title;
	}

	public String getFileName() {
		return fileName;
	}

	public void setFileName(String fileName) {
		this.fileName = fileName;
	}

	public int getPixWidth() {
		return pixWidth;
	}

	public void setPixWidth(int width) {
		this.pixWidth = width;
	}

	public int getPixHeight() {
		return pixHeight;
	}

	public void setPixHeight(int height) {
		this.pixHeight = height;
	}

	@Override
	public String getDescription() {
		return getTitle();
	}
}

 3.  創建裝飾類

/**
 * 
 * @author dgm
 * @describe "圖像裝飾,類比Decorator"
 */
public abstract class ImageDecorator extends PhotoImage {
	protected PhotoImage target;
	
	public ImageDecorator(PhotoImage target) {
		this.target = target;
	}
	
	@Override
	public String getDescription() {
		return target.getDescription();
	}
	
	@Override
	public String toString() {
		return getDescription();
	}
}

4.  擴寫具體實現的裝飾類(上色,裝框,列印,其實還接著加,比如售賣)

ColorDecorator.java、FrameDecorator.java、PrintDecorator.java

public class ColorDecorator extends ImageDecorator {
	String color;
	
	public ColorDecorator(String color, PhotoImage target) {
		super(target);
		this.color = color;
	}

	@Override
	public String getDescription() {
		return target.getDescription() + ", 上色(" + color + ")";
	}
}

public class FrameDecorator extends ImageDecorator {
	private String desc;
	public FrameDecorator(String desc, PhotoImage target) {
		super(target);
		this.desc = desc;
	}

	@Override
	public String getDescription() {
		return target.getDescription() + ", "+desc+", ";
	}
}

/**
 * 
 * @author dgm
 * @describe "列印圖像,類比ConcreteDecorator"
 */
public class PrintDecorator extends ImageDecorator {
	private String desc;
	private double printWidth, printHeight;
	
	public PrintDecorator(String desc,double printWidth, double printHeight, PhotoImage target) {
		super(target);
		this.desc = desc;
		this.printWidth = printWidth;
		this.printHeight = printHeight;
	}

	@Override
	public String getDescription() {
		return target.getDescription() + desc+ " " + String.format("(%4.1f x %4.1f in)", getPrintWidth(), getPrintHeight());
	}

	public double getPrintWidth() {
		return printWidth;
	}

	public void setPrintWidth(double printWidth) {
		this.printWidth = printWidth;
	}

	public double getPrintHeight() {
		return printHeight;
	}

	public void setPrintHeight(double printHeight) {
		this.printHeight = printHeight;
	}
}

 

5. 測試圖像裝飾效果:

/**
 * 
 * @author dgm
 * @describe "裝飾模式(裝飾的是圖片圖像)測試"
 */
public class PhotoImageDecoratorTest {

	public static void main(String[] args) {
		//先生成圖像對象,然後著色,裝訂,最後列印
		PrintDecorator image = new PrintDecorator("彩色激光列印要高清", 19, 11,
				new FrameDecorator("鑲金裝框",new ColorDecorator("天空式藍色", new PhotoImageWrapper(
						"蒙娜麗莎的微笑", "1968/ifd.00042.jpg"))));
		System.out.println(image);
	}

	private static void addToPrintOrder(PrintDecorator image) {
		System.out.println(image);
	}
}

輸出效果: 

只有0是原材料,123修飾

 入庫https://github.com/dongguangming/design-pattern/tree/master/src/code/decorator

案例: 比如原生jdk中io流就用到了裝飾模式,裝飾一個流

        InputStream inputStream = new FileInputStream("f:\\InstallCert.java");
        InputStream bis = new BufferedInputStream(inputStream);
        InputStream in = new DataInputStream(bis);
        InputStreamReader isr = new InputStreamReader(in);
        BufferedReader br  = new BufferedReader(isr);
        String line = null;
        while ((line = br.readLine()) != null) {
            //將文本列印到控制台
            System.out.println(line);
        }

 鏈式寫法長

BufferedReader br = new BufferedReader(new InputStreamReader(new DataInputStream(
        new BufferedInputStream(new FileInputStream("f:\\InstallCert.java")))));

 和javafx.swing 開發中component的擴展開發也都用到了裝飾模式

MyBatis框架中cache也有裝飾的影子

Spring也有裝飾,發揮的更牛叉(沒有Bean就沒有其他擴展):BeanDefinitionDecorator(要分開理解,BeanDefinition和Decorator),是Bean定義階段的裝飾,當然還有其他裝飾比如裝飾WebSocket 對象。

 

裝飾模式,特別於日誌、緩存、io流等場景太合適不過了!!!

 

現實生活中裝飾模式也發揮的很好,比如汽車需要裝飾(洗車、音響加裝、拋光、打蠟、劃痕處理、環保桑拿、地板鋪設、貼防爆膜、烤前後擋膜、中控門鎖、輪胎動平衡、四輪定位、倒車雷達、防盜器等)、房子也要裝飾(鋪地板、地暖、傢具、陽臺要不要種點花)、飯店開業經營(門口要擺兩排花、室內安裝液晶影視、買了紅色鯉魚加水藻裝飾下)。。。太多了!!!

特別註意它和其他幾種模式的區別:

 

小結:  如果你想給原生對象(比如車、房、飯店,其實主題就一個)添加額外的行為職責,又不想變它,就可以使用該模式,那就裝飾它原生對象, 使用場景頗多,具體自己發揮去,走自己的路,然別人去說吧

 

參考:

0. 裝飾模式  https://baike.baidu.com/item/%E8%A3%85%E9%A5%B0%E6%A8%A1%E5%BC%8F/10158540

1.  如何使用Decorator模式 http://www.uml.org.cn/sjms/20041655.htm

2.  javax.swing.JScrollPane  https://docs.oracle.com/javase/8/docs/api/javax/swing/JScrollPane.html

 

3. Decorator Design Pattern Applied https://www.javacodegeeks.com/2014/08/decorator-design-pattern-applied.html

4.  The Decorator Pattern in Depth  https://blogs.oracle.com/javamagazine/the-decorator-pattern-in-depth

5.  Decorator Design Pattern in Java with Example  https://www.java67.com/2013/07/decorator-design-pattern-in-java-real-life-example-tutorial.html?m=1


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 多級指針 多級指針案例1 多級指針案列2 靜態變數不能跨函數使用 1. 記憶體越界:程式訪問了一個不該被訪問的記憶體,函數內的靜態空間,不能被其他函數調用訪問。 2. 函數中的記憶體空間,隨函數終止而被釋放。記憶體空間釋放後的內容不屬於其他函數,其他函數無許可權訪問。但釋放後的記憶體空間的地址是可以被其他函數讀 ...
  • 一、初衷 本來要肝演算法的還是私下搞吧,但是在某通信公司不會C,幹活都沒法乾 所以 肝 肝 肝 肝 肝 肝 肝,不會很詳細,但是只要寫出來就是我不會的,或者需要總結的,如下牛*的參考書,比大學課本譚浩強​好多了。​ 二、高效性 C語言居右通常是彙編語言才有的微調控功能,彙編語言是為特殊中央處理單元設計 ...
  • 自己用C語言實現的推箱子的游戲,在寫這個的期間瀏覽,查看了許多的博客和論壇。(寫於大一下學期) 這個游戲我用的是VS2010和EasyX圖形庫寫的。 如有錯誤,望指正。 代碼在最後。 游戲的效果圖 游戲界面 通關界面 這個3.0是因為,有過2次大的修改。 還有這個時間的數字是不動的,這裡不太懂怎麼弄 ...
  • 1 問題 對 排序,只要在 後面加欄位就可以了,可以通過加 或`asc`來選擇降序或升序。但排序規則是預設的,數字、時間、字元串等都有自己預設的排序規則。有時候需要按自己的想法來排序,而不是按欄位預設排序規則。 比如欄位值為英文欄位: 、`Tuesday Wednesday`等,如果按欄位預設排序規 ...
  • 項目簡介 項目來源於: "https://gitee.com/zzdoreen/SSMS" 本系統基於 JSP+Servlet+Mysql 一個基於JSP+Servlet+Jdbc的學生成績管理系統。涉及技術少,易於理解,適合 JavaWeb初學者 學習使用。 難度等級:入門 技術棧 編輯器 Ecl ...
  • #! python3import random ''' 目標:製作N份選項無序的試卷 步驟:1.創建文件(試卷文件和對應答案文件) 2.寫入題頭 3.寫入題目和選項 4.關閉文件 重點:1.無序選項如何實現 已有數據是字典形式,key是題目內容,對應的value是正確答案, 選項都是value,所以 ...
  • scala基礎 安裝scala(不推薦使用最新版本,2.11.x夠用了) "scala官網" "2.11.12版本下載頁面" 這裡我選擇2.11.12版本,在下載頁面往下拉,選擇scala 2.11.12.msi(windows用戶),msi安裝比較簡單,一直點點就行。如果下載速度慢,建議用迅雷。 ...
  • 今天學習了一下集合類的知識,練習Set的時候發現 1*Set集合允許重覆的值插入,但是重覆的值會被覆蓋; 2*List集合允許重覆的值插入,但是重覆的值會不會被覆蓋; import java.util.*; public class Test14702 { public static void ma ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...