责任链(Chain of Responsibility)模式用于让多个对象可来处理单个请求之情況,其運作原则如下所示:
- 存在一个对象链(如直链表、树或任何其他数据结构)。
- 客戶端将请求发送给链中的第一个对象。
- 某一对象可决定是否处理该请求,若無則将请求转发给下一对象,重复此过程直到链尾。
對客户端之請求仅需要知道如何与链的起始节点处理元素进行通信即可,而不須拥有处理元素的所有引用,但每个处理元素要知道其連接的下一个后继者,並不須知道其它处理元素,这通常是一种单向链表,这种链式组织方式之最大用处:可解耦发送方(*客户端)和接收方(*处理元素)。通过使用责任链模式,我们能让许多不同对象来处理一个特定请求,在我们预先不知道应该由哪个对象来处理某个请求时,这是非常有用的。
Apple的Cocoa 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
總结:
在无法预先知道处理程序时,可構建責任鏈模式來对请求作处理,在此模式中发送方只須访问链中的首个节点,若首个节点不能处理请求,则转发交给下一个节点,依此循環直到请求被某个节点处理或直至遍历链结束為止,这种设计可实现发送方与接收方之间的解耦。