大話設計模式:第24章 職責鏈模式

第24章:職責鏈模式

職責鏈模式

職責鏈(chain of responsibility)模式:使多個對象都有機會處理請求,從而避免請求的發送者和接收者之間的耦合關係。將這個對象連成一條鏈,並沿着這條鏈傳遞該請求,直到有一個對象處理它爲止。

在這裏插入圖片描述

Handler類,定義一個處理請示的接口。

在這裏插入圖片描述

ConcreteHandler類,具體處理者類,處理它所負責的請求,可訪問它的後繼者,如果可處理該請求,就處理之,否則就將該請求轉發給它的後繼者。

ConcreteHandler1,當請求數在0到10之間則有權處理,否則轉到下一位。

在這裏插入圖片描述

ConcreteHandler2,當請求數在10到20之間則有權處理,否則轉到下一位。

在這裏插入圖片描述

ConcreteHandler3,當請求數在20到30之間則有權處理,否則轉到下一位。

在這裏插入圖片描述

客戶端代碼,向鏈上的具體處理者對象提交請求。

在這裏插入圖片描述

職責鏈的好處

當客戶提交-一個請求時,請求是沿鏈傳遞直至有一個ConcreteHandler對象負責處理它。

接收者和發送者都沒有對方的明確信息,且鏈中的對象自己也並不知道鏈的結構。結果是職責鏈可簡化對象的相互連接,它們僅需保持一個指向其後繼者的引用,而不需保持它所有的候選接受者的引用,大大降低了耦合度。

由於是在客戶端來定義鏈的結構,也就是可以隨時地增加或修改處理一個請求的結構,增強了給對象指派職責的靈活性。

注意:一個請求極有可能到了鏈的末端都得不到處理,或者因爲沒有正確配置而得不到處理。

職責鏈模式示例

任務:加薪逐級審批

在這裏插入圖片描述

from abc import ABCMeta, abstractmethod
from typing import Text
class Request(object):
    """
    申請類
    """
    def __init__(self) -> None:
        # 申請類別
        self.__request_type = None
        # 申請內容
        self.__request_content = None
        # 申請數量
        self.__number = None
        
    @property
    def request_type(self) -> Text:
        return self.__request_type
    @request_type.setter
    def request_type(self, value: Text) -> None:
        self.__request_type = value
        
    @property
    def request_content(self) -> Text:
        return self.__request_content
    @request_content.setter
    def request_content(self, value: Text) -> None:
        self.__request_content = value
        
    @property
    def number(self) -> Text:
        return self.__number
    @number.setter
    def number(self, value: int) -> None:
        self.__number = value
        
class Manager(metaclass=ABCMeta):
    """
    管理者(Handler)類
    """
    def __init__(self, name: Text) -> None:
        self._name = name
        self._superior = None
        
    def set_superior(self, superior: object) -> None:
        """
        設置管理者的上級
        """
        self._superior = superior
        
    @abstractmethod
    def request_applications(self, request: Request) -> None:
        """
        申請請求
        """
        pass
    
class CommonManager(Manager):
    """
    經理類
    """
    def __init__(self, name: Text) -> None:
        super(CommonManager, self).__init__(name)
        
    def request_applications(self, request: Request) -> None:
        """
        經理權限
        """
        if (request.request_type == "請假") and (request.number <= 2):
            print("{}:{},數量{},被批准".format(self._name, request.request_type, request.number))
        else:
            if self._superior:
                self._superior.request_applications(request)
                
class Majordomo(Manager):
    """
    總監類
    """
    def __init__(self, name: Text) -> None:
        super(Majordomo, self).__init__(name)
        
    def request_applications(self, request: Request) -> None:
        """
        總監權限
        """
        if (request.request_type == "請假") and (request.number <= 5):
            print("{}:{},數量{},被批准".format(self._name, request.request_type, request.number))
        else:
            if self._superior:
                self._superior.request_applications(request)
                
class GeneralManager(Manager):
    """
    總經理類
    """
    def __init__(self, name: Text) -> None:
        super(GeneralManager, self).__init__(name)
        
    def request_applications(self, request: Request) -> None:
        """
        經理權限
        """
        if (request.request_type == "請假"):
            print("{}:{},數量{},被批准".format(self._name, request.request_type, request.number))
        elif (request.request_type == "加薪") and (request.number <= 500):
            print("{}:{},數量{},被批准".format(self._name, request.request_type, request.number))
        else:
            print("{}:{},數量{},再說吧".format(self._name, request.request_type, request.number))
                
# 客戶端調用

if __name__ == "__main__":
    
    jinli = CommonManager("金利")
    zongjian = Majordomo("宗劍")
    zhongjingli = GeneralManager("鍾精勵")
    jinli.set_superior(zongjian)
    zongjian.set_superior(zhongjingli)
    
    request = Request()
    request.request_type = "請假"
    request.request_content = "小菜請假"
    request.number = 1
    jinli.request_applications(request)
    
    request.number = 4
    jinli.request_applications(request)
    
    request.request_type = "加薪"
    request.request_content = "小菜加薪"
    request.number = 500
    jinli.request_applications(request)
    
    request.number = 1000
    jinli.request_applications(request)
    
金利:請假,數量1,被批准
宗劍:請假,數量4,被批准
鍾精勵:加薪,數量500,被批准
鍾精勵:加薪,數量1000,再說吧
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章