news 2026/5/1 10:33:45

Python 命令模式

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
Python 命令模式

Python 中的命令模式(Command Pattern)

命令模式是一种行为型设计模式,其核心目的是:
将一个请求(操作)封装为一个对象,从而让你可以参数化客户端的不同请求、将请求排队或记录请求日志,还支持可撤销的操作

形象比喻:就像餐厅点餐——顾客(调用者)向服务员(调用者)下订单(命令对象),服务员把订单交给厨师(接收者)执行。订单可以被保存、排队、甚至取消。

命令模式的优点
  • 解耦调用者和接收者:调用者不需要知道谁来执行命令
  • 支持撤销/重做(Undo/Redo):最经典的应用
  • 支持命令队列、延迟执行、日志记录
  • 易于扩展新命令:符合开闭原则
  • 可以组合命令(宏命令)
典型应用场景
  • GUI 按钮、菜单操作(点击执行命令)
  • 撤销/重做功能(文本编辑器、绘图软件)
  • 事务系统(数据库操作回滚)
  • 任务队列、线程池
  • 宏录制(一系列命令组合执行)
  • 远程控制、日志记录
Python 实现示例:带撤销功能的文本编辑器
fromabcimportABC,abstractmethodfromtypingimportList# 接收者(Receiver):真正执行操作的对象classTextEditor:def__init__(self,text:str=""):self.text=textdefinsert(self,position:int,content:str):self.text=self.text[:position]+content+self.text[position:]print(f"插入文本: '{content}' → 当前文本: '{self.text}'")defdelete(self,start:int,end:int):deleted=self.text[start:end]self.text=self.text[:start]+self.text[end:]print(f"删除文本: '{deleted}' → 当前文本: '{self.text}'")returndeleted# 返回被删除的内容,用于撤销def__str__(self):returnself.text# 命令接口(Command)classCommand(ABC):@abstractmethoddefexecute(self):pass@abstractmethoddefundo(self):pass# 具体命令:插入命令classInsertCommand(Command):def__init__(self,editor:TextEditor,position:int,content:str):self.editor=editor self.position=position self.content=contentdefexecute(self):self.editor.insert(self.position,self.content)defundo(self):# 撤销插入就是删除刚插入的内容self.editor.delete(self.position,self.position+len(self.content))# 具体命令:删除命令classDeleteCommand(Command):def__init__(self,editor:TextEditor,start:int,end:int):self.editor=editor self.start=start self.end=end self.deleted_content=None# 执行时保存被删内容defexecute(self):self.deleted_content=self.editor.delete(self.start,self.end)defundo(self):ifself.deleted_contentisnotNone:self.editor.insert(self.start,self.deleted_content)# 调用者(Invoker):管理命令历史,支持撤销classCommandManager:def__init__(self):self.history:List[Command]=[]self.redo_stack:List[Command]=[]defexecute_command(self,command:Command):command.execute()self.history.append(command)self.redo_stack.clear()# 新命令后,清空重做栈defundo(self):ifnotself.history:print("没有可撤销的操作")returncommand=self.history.pop()command.undo()self.redo_stack.append(command)print("撤销成功")defredo(self):ifnotself.redo_stack:print("没有可重做的操作")returncommand=self.redo_stack.pop()command.execute()self.history.append(command)print("重做成功")# 客户端使用if__name__=="__main__":editor=TextEditor("Hello World")manager=CommandManager()print(f"初始文本: '{editor}'\n")# 执行插入insert_cmd=InsertCommand(editor,5,", Python")manager.execute_command(insert_cmd)# 输出: Hello, Python World# 执行删除delete_cmd=DeleteCommand(editor,5,13)# 删除 ", Python"manager.execute_command(delete_cmd)# 输出: Hello Worldprint("\n--- 开始撤销 ---")manager.undo()# 撤销删除 → 恢复 ", Python"manager.undo()# 撤销插入 → 回到 "Hello World"print("\n--- 开始重做 ---")manager.redo()# 重做插入manager.redo()# 重做删除

输出示例

初始文本: 'Hello World' 插入文本: ', Python' → 当前文本: 'Hello, Python World' 删除文本: ', Python' → 当前文本: 'Hello World' --- 开始撤销 --- 撤销成功 # 恢复到 Hello, Python World 撤销成功 # 恢复到 Hello World --- 开始重做 --- 重做成功 # 插入 → Hello, Python World 重做成功 # 删除 → Hello World
Pythonic 简化版:函数作为命令

当命令简单时,可以用 callable(函数/闭包)代替类:

classSimpleCommandManager:def__init__(self):self.history=[]self.redo_stack=[]defexecute(self,do_func,undo_func):do_func()self.history.append((do_func,undo_func))self.redo_stack.clear()defundo(self):ifself.history:do_func,undo_func=self.history.pop()undo_func()self.redo_stack.append((do_func,undo_func))defredo(self):ifself.redo_stack:do_func,undo_func=self.redo_stack.pop()do_func()self.history.append((do_func,undo_func))# 使用text="Hello"manager=SimpleCommandManager()definsert():nonlocaltext text+=" World"print("→",text)defremove():nonlocaltext text=text[:-6]print("←",text)manager.execute(insert,remove)manager.execute(lambda:print("额外操作"),lambda:None)manager.undo()manager.undo()manager.redo()
命令模式结构总结
角色说明
Command抽象命令接口(execute/undo)
ConcreteCommand具体命令(InsertCommand、DeleteCommand)
Receiver接收者(TextEditor)
Invoker调用者(CommandManager)
Client创建命令并设置到调用者
命令模式 vs 其他模式对比
模式目的是否支持撤销典型场景
命令封装请求,支持撤销/队列编辑器、事务、宏
策略算法可替换支付、排序
观察者状态变化广播通知事件监听
责任链请求沿链传递处理流程
Python 中的实用建议
  • 简单命令:用 lambda 或函数闭包(最 Pythonic)
  • 复杂命令:用类实现,支持状态保存
  • 宏命令:创建一个 CompositeCommand,内部持有多个子命令
  • 常与观察者模式结合(命令执行后通知 UI 更新)
classMacroCommand(Command):def__init__(self,commands:List[Command]):self.commands=commandsdefexecute(self):forcmdinself.commands:cmd.execute()defundo(self):forcmdinreversed(self.commands):cmd.undo()

命令模式是实现可撤销操作操作历史的基石,在编辑器、游戏存档、事务系统中广泛应用。

如果你想看更多实战例子(如 GUI 按钮命令、线程池任务队列、宏录制、结合队列的延迟执行),或者如何与备忘录模式结合实现多级撤销,欢迎继续问!

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/5/1 4:14:09

错过等一年!Open-AutoGLM 2.0正式上线GitHub,这些功能你必须掌握

第一章:错过等一年!Open-AutoGLM 2.0正式发布Open-AutoGLM 2.0 正式上线,标志着自动化大模型应用开发迈入全新阶段。该版本在性能、易用性和扩展性方面实现全面升级,专为开发者与企业用户打造高效、灵活的AI解决方案构建平台。核心…

作者头像 李华
网站建设 2026/5/1 5:23:36

破解“写作围城”:当期刊投稿遇上行家级AI协作者

文献迷雾中不再焦虑,智能工具重构写作全流程的效率与质量深夜的实验室,屏幕上摊着十几个窗口——文献PDF、草稿文档、数据表格和格式混乱的参考文献列表,学者们正试图从数字碎片中拼凑论文的完整形态,这种场景几乎成为科研通病。传…

作者头像 李华
网站建设 2026/5/1 5:24:15

12、GAN技术:从渐进式生成到半监督学习的突破

GAN技术:从渐进式生成到半监督学习的突破 1. 渐进式生成对抗网络(Progressive GAN)的实际应用 1.1 医学影像合成的卓越成果 在医学领域,研究人员利用大量的医学乳腺X光片数据集,借助渐进式生成对抗网络(Progressive GAN,简称PGGAN)技术,成功生成了分辨率高达1280 …

作者头像 李华
网站建设 2026/5/1 5:21:37

17、CycleGAN与对抗样本:原理、训练与应用

CycleGAN与对抗样本:原理、训练与应用 1. CycleGAN概述 CycleGAN是一种强大的图像到图像转换模型,它能够在无需配对图像数据的情况下,实现不同领域之间的图像转换,例如将苹果转换为橙子,反之亦然。下面我们将详细介绍CycleGAN的构建、训练和应用。 1.1 构建生成器 生成…

作者头像 李华
网站建设 2026/5/1 5:21:55

18、对抗样本:从原理到防御的全面解析

对抗样本:从原理到防御的全面解析 1. 训练数据的挑战 在处理图像数据时,即使是同一类别的图像,当拍摄角度稍有变化,它们之间的差异也可能很大。以一个包含100,000个300300的RGB图像的训练集为例,我们需要处理270,000个维度的数据。当考虑所有可能的图像(而非实际观察到…

作者头像 李华