初識設計模式 - 職責鏈模式

来源:https://www.cnblogs.com/fatedeity/archive/2022/11/08/16868497.html
-Advertisement-
Play Games

職責鏈模式是一種與策略模式類似的設計模式,都是使用多個對象去處理同一個請求。不同的是,職責鏈模式針對的一條鏈路上的所有對象,而不是“非此即彼”的關係。 ...


簡介

職責鏈設計模式(Chain Of Responsibility Design Pattern)的定義是,將請求的發送和接收解耦,讓多個接收對象都有機會處理這個請求。

同時,將這些接收對象串成一條鏈,並沿著這條鏈傳遞這個對象,直至鏈上的某個接收對象能夠處理這個請求為止。

職責鏈可以是一條直線、一個環或一個樹形結構,最常見的職責鏈是直線型,即沿著一條單向的鏈來傳遞請求。

典型實現

首先,定義一個抽象處理者 Handler 類,其代碼示例如下:

public abstract class Handler {
    // 維持對下一個處理者的引用
    protected Handler successor;

    public void setHandler(Handler successor) {
        this.successor = successor;
    }

    public abstract void handleRequest(String Request);
}

然後,定義一個具體處理者 ConcreteHandler 子類,其代碼示例如下:

public class ConcreteHandler extends Handler {
    public void handleRequest(String request) {
        // 處理請求或者轉發請求
        this.successor.handleRequest(request);
        // 執行完當前處理方法後,可以執行下一個處理者的處理,完成鏈路迴圈
    }
}

對於客戶端而言,只需要知道第一個具體處理者是誰即可,無需關心後續的其他處理者。這就像是操作鏈表一樣,知道鏈表的頭結點即可訪問鏈表的所有結點。

分類

根據處理者對象的行為,職責鏈模式可以分為純的職責鏈模式和不純的職責鏈模式。

純的職責鏈模式

一個純的職責鏈模式要求一個具體處理者對象只能在兩種行為中選擇一個:要麼承擔全部責任,要麼將責任推給下家。

同時,純的職責鏈模式要求一個請求必須被某一個具體處理者對象所接受,不能出現某個請求未被處理者對象接收的情形。

不純的職責鏈模式

不純的職責鏈模式是與純的職責鏈模式相對的一種模式。

在一個不純的職責鏈模式中,允許某個請求被具體處理者部分處理後還能向下傳遞,或者一個具體處理者處理完某個請求後其後繼處理者可以繼續處理該對象,而且同一個請求可以最終不被任何處理者對象所接收。

總結

優點

職責鏈模式的主要優點如下:

  • 將發送者和接收者解耦,客戶端無需知道請求被哪一個對象處理
  • 當工作流程發生變化,可以動態地改變鏈內的成員或調動它們的次序,也可動態的新增或刪除職責
  • 通過鏈式結構串聯處理者,可以根據需要增加新的處理類,符合開閉原則
  • 純的職責鏈模式明確了各類的職責範圍,符合類的單一職責原則

缺點

職責鏈模式的主要缺點如下:

  • 由於請求沒有一個明確的處理者,不能保證請求一定會被處理
  • 對於較長的職責鏈,請求的處理涉及到多個處理對象,系統性能將受到一定影響
  • 職責鏈的建立要靠客戶端來保證,增加了客戶端的複雜性,建鏈不當可能造成迴圈引用

適用場景

職責鏈模式的適用場景如下:

  • 多個對象可以處理一個請求,但具體由哪一個對象處理在運行時自動確定
  • 需要在不明確指定請求處理者的情況下,向多個處理者中的一個提交請求
  • 動態地指定一組處理者,或者改變鏈中處理者之間的次序

源碼

在 JDK 中 java.util.logging.Logger 記錄日誌有可能有多個不同的 Handler 處理器,如果使用這些 Handler 處理器就是一種職責鏈模式的運用。

如下是源碼部分:

public void log(LogRecord record) {
    if (!isLoggable(record.getLevel())) {
        return;
    }
    Filter theFilter = config.filter;
    if (theFilter != null && !theFilter.isLoggable(record)) {
        return;
    }

    Logger logger = this;
    while (logger != null) {
        final Handler[] loggerHandlers = isSystemLogger
            ? logger.accessCheckedHandlers()
            : logger.getHandlers();

        for (Handler handler : loggerHandlers) {
            handler.publish(record);
        }

        final boolean useParentHdls = isSystemLogger
            ? logger.config.useParentHandlers
            : logger.getUseParentHandlers();

        if (!useParentHdls) {
            break;
        }

        logger = isSystemLogger ? logger.parent : logger.getParent();
    }
}

首發於翔仔的個人博客,點擊查看更多。


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

-Advertisement-
Play Games
更多相關文章
  • 1、什麼是CSS Cascading Style Sheet 級聯樣式表。 表現HTML或XHTML文件樣式的電腦語言。 包括對字體、顏色、邊距、高度、寬度、背景圖片、網頁定位等設定。 2、CSS的發展史 CSS1.0 讀者可以從其他地方去使用自己喜歡的設計樣式去繼承性地使用樣式; CSS2.0 ...
  • 虛擬DOM就是用JS來模擬DOM結構的,它並不是真正的DOM。 為什麼使用虛擬DOM? 用傳統的方式去操作DOM的時候,瀏覽器會從構建DOM樹開始,從頭到尾執行一遍流程。簡單來說,就是會觸發重排與重繪。 比如說,在一次操作中,需要更新10個DOM節點。 理想是一次性構建完成DOM樹,但是瀏覽器並不會 ...
  • 一、事件委派 1.理解DOM事件流 事件流描述的是從頁面中接收事件的順序。事件發生時會在元素節點之間按照特定的順序傳播,這個傳播過程即DOM事件流。 DOM事件流分為三個階段: 捕獲階段:從上往下 當前目標階段 冒泡階段:從下往上 事件流如下圖所示: 註意事項: JavaScript代碼中只能執行捕 ...
  • 一.商品列表 1.1 獲取數據 首先能夠進入商品列表的途徑 傳的數據有 瞭解了這個之後就可以開始了,先創建分支 創建編譯模式,並分配初試數據 這個時候就可以獲取數據了 需要的數據 所以在發起請求之前需要整理一下數據,先定義數據 整理數據發起請求 1.2 渲染頁面 ==註意我們可以去定義一個預設的圖片 ...
  • 本文主要記錄 Vue.js 中的 router 管理,涉及如何使用路由實現單頁面應用(SPA)的組件切換,以及路由的一些 API 操作。 ...
  • 1、表單語法 method: 規定如何發送表單數據常用值:get post 在實際網頁開發中通常採用post方式提交表單數據 action: 表示向何處發送表單數據 <form method="post" action="result.html"> <p>名字:<input name="name" ...
  • 1、列表 無序列表和定義列表在網頁製作中應用非常廣泛 什麼是列表: 列表就是信息資源的一種展示形式。它可以使信息結構化和條理化,並以列表的樣式顯示出來,以便瀏覽者能更快捷地獲得相應的信息。 無序列表 <!--ul 聲明無序列表--> <ul> <!--li 聲明列表項--> <li>語文</li> ...
  • 面向對象之魔法方法 一、魔法方法的概念 ​ 定義在類中的雙下方法都可以稱為魔法方法 ​ 不需要人為調用,在特定的條件下會自動觸發,並運行 ​ 類似於__ init__, 當我在使用類產生對象時,會自動觸發 class Foo: # 雙下init就是魔法方法的一種 def __init__(self, ...
一周排行
    -Advertisement-
    Play Games
  • 在C#中使用SQL Server實現事務的ACID(原子性、一致性、隔離性、持久性)屬性和使用資料庫鎖(悲觀鎖和樂觀鎖)時,你可以通過ADO.NET的SqlConnection和SqlTransaction類來實現。下麵是一些示例和概念說明。 實現ACID事務 ACID屬性是事務處理的四個基本特征, ...
  • 我們在《SqlSugar開發框架》中,Winform界面開發部分往往也用到了自定義的用戶控制項,對應一些特殊的界面或者常用到的一些局部界面內容,我們可以使用自定義的用戶控制項來提高界面的統一性,同時也增強了使用的便利性。如我們Winform界面中用到的分頁控制項、附件顯示內容、以及一些公司、部門、菜單的下... ...
  • 在本篇教程中,我們學習瞭如何在 Taurus.MVC WebMVC 中進行數據綁定操作。我們還學習瞭如何使用 ${屬性名稱} CMS 語法來綁定頁面上的元素與 Model 中的屬性。通過這些步驟,我們成功實現了一個簡單的數據綁定示例。 ...
  • 是在MVVM中用來傳遞消息的一種方式。它是在MVVMLight框架中提供的一個實現了IMessenger介面的類,可以用來在ViewModel之間、ViewModel和View之間傳遞消息。 Send 接受一個泛型參數,表示要發送的消息內容。 Register 方法用於註冊某個對象接收消息。 pub ...
  • 概述:在WPF中,通過EventHandler可實現基礎和高級的UI更新方式。基礎用法涉及在類中定義事件,併在UI中訂閱以執行更新操作。高級用法藉助Dispatcher類,確保在非UI線程上執行操作後,通過UI線程更新界面。這兩種方法提供了靈活而可靠的UI更新機制。 在WPF(Windows Pre ...
  • 概述:本文介紹了在C#程式開發中如何利用自定義擴展方法測量代碼執行時間。通過使用簡單的Action委托,開發者可以輕鬆獲取代碼塊的執行時間,幫助優化性能、驗證演算法效率以及監控系統性能。這種通用方法提供了一種便捷而有效的方式,有助於提高開發效率和代碼質量。 在軟體開發中,瞭解代碼執行時間是優化程式性能 ...
  • 概述:Cron表達式是一種強大的定時任務調度工具,通過配置不同欄位實現靈活的時間規定。在.NET中,Quartz庫提供了簡便的方式配置Cron表達式,實現精準的定時任務調度。這種靈活性和可擴展性使得開發者能夠根據需求輕鬆地制定和管理定時任務,例如每天備份系統日誌或其他重要操作。 Cron表達式詳解 ...
  • 概述:.NET提供多種定時器,如System.Windows.Forms.Timer適用於UI,System.Web.UI.Timer用於Web,System.Diagnostics.Timer用於性能監控,System.Threading.Timer和System.Timers.Timer用於一般 ...
  • 問題背景 有同事聯繫我說,在生產環境上,訪問不了我負責的common服務,然後我去檢查common服務的health endpoint, 沒問題,然後我問了下異常,timeout導致的System.OperationCanceledException。那大概率是客戶端的問題,會不會是埠耗盡,用ne ...
  • 前言: 在本篇 Taurus.MVC WebMVC 入門開發教程的第四篇文章中, 我們將學習如何實現數據列表的綁定,通過使用 List<Model> 來展示多個數據項。 我們將繼續使用 Taurus.Mvc 命名空間,同時探討如何在視圖中綁定並顯示一個 Model 列表。 步驟1:創建 Model ...