行為型模式--责任链模式

 

责任链(Chain of Responsibility)模式用于让多个对象可来处理单个请求之情況,其運作原则如下所示:

  • 存在一个对象链(如直链表、树或任何其他数据结构)。
  • 客戶端将请求发送给链中的第一个对象
  • 某一对象可决定是否处理该请求,若無則将请求转发给下一对象,重复此过程直到链尾。

對客户端之請求仅需要知道如何与链的起始节点处理元素进行通信即可,而不須拥有处理元素的所有引用,但每个处理元素要知道其連接的下一个后继者,並不須知道其它处理元素,这通常是一种单向链表,这种链式组织方式之最大用处:可解耦发送方(*客户端)和接收方(*处理元素)。通过使用责任链模式,我们能让许多不同对象来处理一个特定请求,在我们预先不知道应该由哪个对象来处理某个请求时,这是非常有用的。

AppleCocoa Touch框架是使用责任链来处理事件,如按钮手势,當在某个视图接收到一个無法处理的事件时,則会将該事件转发给其上之父類视图,依此循環直到有视图能够处理这个事件至视图链结束為止,这在基于事件的编程中是常見的設計手法,如一次鼠标左击之单个事件可被事件监听者捕获以處理。

示例:

class Event:                          #事件封裝類

    def __init__(self, name):

        self.name = name

 

    def __str__(self):     #打印時要取得之字串資訊

        return self.name

 

class Widget:             #控件抽象類

    def __init__(self, parent=None):     #重點:parent是用來指定鏈中之上層類,若沒提供則表示無上層類

        self.parent = parent

 

    def handle(self, event):             #對傳入event參數作動態分發

        """

        此方法主要控制动态分发,使用hasattr()和getattr()來决定特定事件要如何被处理,當目前控件無法处理時且有指定parent,则交由parent之handle()方法處理之;若無指定parent,則交由handle_default()方法處理之。

        """

        handler = 'handle_{}'.format(event)        #組合格式字串(*此要配合可處理程序來命名)

        if hasattr(self, handler):

            getattr(self, handler)(event)             #傳入event參數給處理元素

        elif self.parent:                                                #指定其上層類

            self.parent.handle(event)

        elif hasattr(self, 'handle_default'):

            self.handle_default(event)

 

class MainWindow(Widget):            #控件具体類1

    """

    僅可处理close和default事件

    """

    def handle_close(self, event):

        print('MainWindow: {}'.format(event))

 

    def handle_default(self, event):

        print('MainWindow Default: {}'.format(event))

 

class SendDialog(Widget):

    """

    仅可处理paint事件

    """

    def handle_paint(self, event):

          print('SendDialog: {}'.format(event))

 

class MsgText(Widget):

    """

    仅可处理down事件

    """

    def handle_down(self, event):

        print('MsgText: {}'.format(event))

 

def main():

    """

    構建責任鏈:要注意其中控件的父子关系,此處SendDialog实例的父对象是MainWindow实例,MsgText实例是以SendDialog作为父对象。

    """

    mw = MainWindow()       #這是為何之前parent預設值為None之原因(*表示此類無上層類)

    sd = SendDialog(mw)       #parent是MainWindow

    msg = MsgText(sd)            #parent是SendDialog

          #指定不同事件作測試

    for e in ('down', 'paint', 'unhandled', 'close'):

        evt = Event(e)

          #Test process

        print('Sending event -{}- to MsgText'.format(evt))

        msg.handle(evt)       #將事件交由起始節點執行

          print()

 

if __name__ == '__main__':

    main()

輸出:

Sending event -down- to MsgText

MsgText: down

 

Sending event -paint- to MsgText

SendDialog: paint

 

Sending event -unhandled- to MsgText

MainWindow Default: unhandled

 

Sending event -close- to MsgText

MainWindow: close

 

總结:

在无法预先知道处理程序时,可構建責任鏈模式來对请求作处理,在此模式中发送方只須访问链中的首个节点,若首个节点不能处理请求,则转发交给下一个节点,依此循環直到请求被某个节点处理或直至遍历链结束為止,这种设计可实现发送方与接收方之间的解耦。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值