【裝飾器設計模式詳解】C/Java/JS/Go/Python/TS不同語言實現

来源:https://www.cnblogs.com/letjs/archive/2023/04/04/17287173.html
-Advertisement-
Play Games

簡介 裝飾器模式(Decorator Pattern)是一種結構型設計模式。將對象放入到一個特殊封裝的對象中,為這個對象綁定新的行為,具備新的能力,同時又不改變其原有結構。 如果你希望在無需修改代碼的情況下即可使用對象,且希望在運行時為對象新增額外的行為,可以使用裝飾模式。或者你用繼承來擴展對象行為 ...


簡介

裝飾器模式(Decorator Pattern)是一種結構型設計模式。將對象放入到一個特殊封裝的對象中,為這個對象綁定新的行為,具備新的能力,同時又不改變其原有結構。

如果你希望在無需修改代碼的情況下即可使用對象,且希望在運行時為對象新增額外的行為,可以使用裝飾模式。或者你用繼承來擴展對象行為的方案難以實現或者根本不可行,你可以使用該模式。

作用

  1. 動態地給一個對象添加一些額外的職責,相比生成子類更為靈活。
  2. 在不想增加很多子類的情況下擴展類的能力,實現強大擴展能力。

實現步驟

  1. 創建一個基礎工具介面或抽象類,設定基本的方法。
  2. 增加具體工具類實現基礎介面,保持工具類的規範性。
  3. 創建一個裝飾器抽象類,用於裝飾具體工具,聚合基礎工具,同時也可以實現基礎工具的介面。
  4. 增加多個裝飾器類,繼承抽象類,根據需要設定裝飾能力。

UML

 

Java代碼

基礎形狀介面

// Shape.java 基礎形狀介面
public interface Shape {
   void draw();
}

 

具體形狀實現

// Circle.java 具體形狀實現了基礎形狀介面
public class Circle implements Shape {

   @Override
   public void draw() {
      System.out.println("Circle::draw()");
   }
}
// Square.java 具體形狀實現了基礎形狀介面
public class Square implements Shape {

   @Override
   public void draw() {
      System.out.println("Square::draw()");
   }
}

 

抽象裝飾器

// ShapeDecorator.java 抽象裝飾類,是否實現Shape可選
public abstract class ShapeDecorator implements Shape {
// public abstract class ShapeDecorator {
   protected Shape decoratedShape;

   public ShapeDecorator(Shape decoratedShape) {
      this.decoratedShape = decoratedShape;
   }

   public void draw() {
      decoratedShape.draw();
   }
}

 

具體裝飾器

// RedShapeDecorator.java 具體裝飾器1
public class RedShapeDecorator extends ShapeDecorator {

  public RedShapeDecorator(Shape decoratedShape) {
    super(decoratedShape);
  }

  @Override
  public void draw() {
    decoratedShape.draw();
    setRedColor(decoratedShape);
  }

  private void setRedColor(Shape decoratedShape) {
    System.out.println(
      "RedShapeDecorator::setRedColor() " + decoratedShape.getClass().getName()
    );
  }
}
// ShadowShapeDecorator.java 具體裝飾器2
public class ShadowShapeDecorator extends ShapeDecorator {

  public ShadowShapeDecorator(Shape decoratedShape) {
    super(decoratedShape);
  }

  @Override
  public void draw() {
    // decoratedShape.draw();
    setShadow(decoratedShape);
  }

  private void setShadow(Shape decoratedShape) {
    System.out.println(
      "ShadowShapeDecorator::setShadow() " + decoratedShape.getClass().getName()
    );
  }
}

 

測試調用

    /**
     * 裝飾器模式是將一個對象放到一個裝飾器對象中,執行裝飾器類里的方法時,對象的行為能力得到增強。
     * 先聲明具體對象,然後放到裝飾器,得到一個帶有裝飾器的新對象,該對象具備了新的能力。
     */

    // 聲明形狀
    Shape circle = new Circle();
    Shape square = new Square();

    // 增加紅色裝飾
    ShapeDecorator redCircle = new RedShapeDecorator(circle);
    ShapeDecorator redSquare = new RedShapeDecorator(square);
    circle.draw();
    redCircle.draw();
    redSquare.draw();

    // 增加影子裝飾
    ShadowShapeDecorator shadowCircle = new ShadowShapeDecorator(circle);
    ShadowShapeDecorator shadowSquare = new ShadowShapeDecorator(square);
    shadowCircle.draw();
    shadowSquare.draw();

 

Go代碼

基礎形狀介面

// Shape.go 基礎形狀介面
type Shape interface {
  Draw()
  GetName() string
}

 

具體形狀實現

// Circle.go 具體形狀實現了基礎形狀介面
type Circle struct {
}

func (c *Circle) Draw() {
  fmt.Println("Circle::Draw()")
}

func (c *Circle) GetName() string {
  return "Circle"
}

// Square.go 具體形狀實現了基礎形狀介面
type Square struct {
}

func (c *Square) Draw() {
  fmt.Println("Square::Draw()")
}

func (c *Square) GetName() string {
  return "Square"
}

 

抽象裝飾器

// ShapeDecorator.go 抽象裝飾類,是否實現Shape可選
type ShapeDecorator interface {
  Draw()
}

 

具體裝飾器

// RedShapeDecorator.go 具體裝飾器1
type RedShapeDecorator struct {
  DecoratedShape Shape
}

func (r *RedShapeDecorator) Draw() {
  r.DecoratedShape.Draw()
  r.SetRedColor(r.DecoratedShape)
}

func (r *RedShapeDecorator) SetRedColor(decoratedShape Shape) {
  fmt.Println("RedShapeDecorator::setRedColor() " + decoratedShape.GetName())
}
// ShadowShapeDecorator.go 具體裝飾器2
type ShadowShapeDecorator struct {
  DecoratedShape Shape
}

func (s *ShadowShapeDecorator) Draw() {
  // 裝飾器根據需要是否調用形狀的Draw方法
  // s.DecoratedShape.Draw()
  s.SetShadow(s.DecoratedShape)
}

func (s *ShadowShapeDecorator) SetShadow(decoratedShape Shape) {
  fmt.Println("ShadowShapeDecorator::SetShadow() " + decoratedShape.GetName())
}

 

測試調用

  /**
   * 裝飾器模式是將一個對象放到一個裝飾器對象中,執行裝飾器類里的方法時,對象的行為能力得到增強。
   * 先聲明具體對象,然後放到裝飾器,得到一個帶有裝飾器的新對象,該對象具備了新的能力。
   */

  // 聲明形狀
  var circle = new(src.Circle)
  var square = new(src.Square)

  // 增加紅色裝飾
  var redCircle = &src.RedShapeDecorator{
    DecoratedShape: circle,
  }
  var redSquare = &src.RedShapeDecorator{
    DecoratedShape: square,
  }
  circle.Draw()
  redCircle.Draw()
  redSquare.Draw()

  // 增加影子裝飾
  var shadowCircle = &src.ShadowShapeDecorator{
    DecoratedShape: circle,
  }
  var shadowSquare = &src.ShadowShapeDecorator{
    DecoratedShape: square,
  }
  shadowCircle.Draw()
  shadowSquare.Draw()

 

更多語言版本

不同語言實現設計模式:https://github.com/microwind/design-pattern


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

-Advertisement-
Play Games
更多相關文章
  • 1、webpack的作用 模塊打包工具,可以將項目打包成相容瀏覽器的格式,或者打包成手機app等。 它做的事情是,分析你的項目結構,找到JavaScript模塊以及其它的一些瀏覽器不能直接運行的拓展語言(Scss,TypeScript等),並將其轉換和打包為合適的格式供瀏覽器使用 2、node的作用 ...
  • <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <meta http-equiv="X-UA-Compatible" content="IE=edge"> <meta name="viewport" content="wi ...
  • // 註意: 基本數據類型不存在深淺拷貝,只是值傳遞,複合數據類型才有深淺拷貝之說 var obj1 = { name: "吳亦凡" }; var obj2 = obj1; obj1.name = "羅志祥"; // 相當於把obj1的指針複製了一份給了obj2,兩個指針指向了堆記憶體中的一塊記憶體地址 ...
  • 本篇僅以HBuilder X為例,其餘開發軟體同樣適用 這裡兩種方法,使用script標簽直接引入本地的vue.js或使用CDN引入。 ...
  • 這裡給大家分享我在網上總結出來的一些知識,希望對大家有所幫助 CSS3提供了Animation關鍵幀動畫,我們在工作中比較常用。但在寫CSS動畫的時候,其實Animation能實現兩種動畫模式: 補間動畫 設置關鍵幀的初始狀態,然後在另一個關鍵幀改變這個狀態,比如大小、顏色、位置、透明度等,電腦將自 ...
  • Javascript: 網頁可見區域寬: document.body.clientWidth 網頁可見區域高: document.body.clientHeight 網頁可見區域寬: document.body.offsetWidth (包括邊線的寬) 網頁可見區域高: document.body. ...
  • 前端性能優化——圖片優化 一、圖片優化措施 優化圖片是 Web 前端優化的重要一環,因為圖片是 Web 頁面中最耗費帶寬和載入時間的資源之一。以下是一些通過優化圖片來優化 Web 前端的方法: 壓縮圖片:壓縮圖片可以減少圖片的文件大小,從而減少載入時間。 使用矢量圖形:使用矢量圖形(如 SVG)可以 ...
  • 在JavaScript中,我們經常使用requestAnimationFrame、setTimeout、setInterval和setImmediate來控制代碼的執行時機。它們各有特點和適用場景: 1. requestAnimationFrame: requestAnimationFrame主要用 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...