思想:
意圖:
將一個請求封裝為一個對象畔裕,從而使你可用不同的請求對客戶進(jìn)行參數(shù)化耙箍;對請求排隊或記錄請求日志孽查,以及支持可撤消的操作聂受。
適用性:
抽象出待執(zhí)行的動作以參數(shù)化某對象蒿秦,你可用過程語言中的回調(diào)(call back)函數(shù)表達(dá)這種參數(shù)化機(jī)制。所謂回調(diào)函數(shù)是指函數(shù)先在某處注冊蛋济,而它將在稍后某個需要的時候被調(diào)用棍鳖。Command 模式是回調(diào)機(jī)制的一個面向?qū)ο蟮奶娲贰?/p>
在不同的時刻指定、排列和執(zhí)行請求。一個Command對象可以有一個與初始請求無關(guān)的生存期渡处。如果一個請求的接收者可用一種與地址空間無關(guān)的方式表達(dá)镜悉,那么就可將負(fù)責(zé)該請求的命令對象傳送給另一個不同的進(jìn)程并在那兒實現(xiàn)該請求。
支持取消操作医瘫。Command的Excute 操作可在實施操作前將狀態(tài)存儲起來侣肄,在取消操作時這個狀態(tài)用來消除該操作的影響。Command 接口必須添加一個Unexecute操作醇份,該操作取消上一次Execute調(diào)用的效果稼锅。執(zhí)行的命令被存儲在一個歷史列表中×欧祝可通過向后和向前遍歷這一列表并分別調(diào)用Unexecute和Execute來實現(xiàn)重數(shù)不限的“取消”和“重做”矩距。
支持修改日志,這樣當(dāng)系統(tǒng)崩潰時怖竭,這些修改可以被重做一遍剩晴。在Command接口中添加裝載操作和存儲操作,可以用來保持變動的一個一致的修改日志侵状。從崩潰中恢復(fù)的過程包括從磁盤中重新讀入記錄下來的命令并用Execute操作重新執(zhí)行它們赞弥。
用構(gòu)建在原語操作上的高層操作構(gòu)造一個系統(tǒng)。這樣一種結(jié)構(gòu)在支持事務(wù)( transaction)的信息系統(tǒng)中很常見趣兄。一個事務(wù)封裝了對數(shù)據(jù)的一組變動绽左。Command模式提供了對事務(wù)進(jìn)行建模的方法。Command有一個公共的接口艇潭,使得你可以用同一種方式調(diào)用所有的事務(wù)拼窥。同時使用該模式也易于添加新事務(wù)以擴(kuò)展系統(tǒng)。
案例一:
"""
Command
"""
import os
class MoveFileCommand(object):
def __init__(self, src, dest):
self.src = src
self.dest = dest
def execute(self):
self()
def __call__(self):
print('renaming {} to {}'.format(self.src, self.dest))
os.rename(self.src, self.dest)
def undo(self):
print('renaming {} to {}'.format(self.dest, self.src))
os.rename(self.dest, self.src)
if __name__ == "__main__":
command_stack = []
# commands are just pushed into the command stack
command_stack.append(MoveFileCommand('foo.txt', 'bar.txt'))
command_stack.append(MoveFileCommand('bar.txt', 'baz.txt'))
# they can be executed later on
for cmd in command_stack:
cmd.execute()
# and can also be undone at will
for cmd in reversed(command_stack):
cmd.undo()