python设计模式之命令模式的组成、场景、特点、示例
off999 2024-10-27 11:57 45 浏览 0 评论
命令模式(Command Pattern)是一种常用的设计模式,属于行为型模式的一种。它的主要目的是将一个请求封装为一个对象,从而允许用户使用不同的请求、队列或者日志来参数化其他对象。命令模式也支持可撤销的操作。
1 命令模式的组成
命令模式通常包括以下几个角色:
- Command:命令接口,声明执行操作的接口。
- ConcreteCommand:具体命令,实现命令接口,并调用接收者的一个或多个动作来完成具体的命令。会保持一个接受者的引用。它是一个容器并不会执行具体处理
- Invoker:调用者,负责调用命令对象执行请求。
- Receiver:接收者,知道如何实施与执行一个请求相关的操作。
- Client:客户端,创建一个具体命令对象,并设置其接收者。
2 使用场景
命令模式在实际场景中的作用主要体现在以下几个方面:
- 解耦调用者和接收者:
命令模式可以将发出命令的对象和接收以及执行命令的对象解耦。这意味着命令的发送者不需要知道具体是谁执行的命令,也不需要知道命令是如何实现的。这种解耦使得系统更加灵活,易于扩展和修改。 - 命令的排队与执行:
命令模式允许将命令保存在队列中,然后在需要的时候依次执行。这在多线程程序中特别有用,可以在一个线程中排队命令,并在另一个线程中一一执行,有效地实现了线程间的任务调度。 - 支持撤销和重做操作:
命令模式可以通过实现撤销(undo)和重做(redo)命令来支持用户的撤销操作,这在编辑器、图形界面工具等需要提供用户操作回退功能的应用中非常有用。 - 支持操作的宏命令:
通过命令模式,可以将多个命令组合成一个宏命令,然后以单一命令的方式执行。这种方式常用于实现复杂的用户界面操作、事务性操作,或是需要一系列步骤共同完成一个任务的场景。 - 动态调整命令:
由于命令是对象,可以在运行时动态地改变命令的参数,或者完全更换执行的命令。这为动态调整程序的行为提供了极大的灵活性。 - 日志和事务:
命令模式可以用来记录详细的日志,记录用户的具体操作,或者在需要的时候,实现对命令的持久化。在软件崩溃后,这些记录可以用来恢复用户的操作。在事务型系统中,可以用命令模式来实现事务的提交和回滚。
3 特点
- 解耦命令的发出者和执行者:发出请求的对象与接收和执行请求的对象是解耦的。
- 扩展性好:可以很容易地新增命令或者改变命令的执行者。
- 组合命令:可以组合多个命令,实现复杂的功能。
- 支持撤销操作:命令模式可以用来实现命令的撤销和重做。-
4 示例代码
- 用户界面按钮和菜单项:GUI中的每个按钮或菜单项基本上都是一个命令模式的实现。
- 多级撤销功能:通过命令模式实现命令的历史记录,用户可以进行撤销或重做操作。
- 事务型行为:命令模式可以与数据库事务结合使用,用于实现事务的逻辑,其中每个操作都可以看作是一个命令。
- 排队请求:比如工作队列的设计,将请求排队,并在不同的时间点处理它们。
- 复合命令:可以组合多个命令,形成宏命令。
4.1 用户界面按钮和菜单项
一个简单的遥控器操作电灯的开关:
class Command:
def execute(self):
pass
class Light:
"""接收者:电灯,执行与电灯相关的操作"""
def turn_on(self):
print("电灯打开了。")
def turn_off(self):
print("电灯关闭了。")
class TurnOnCommand(Command):
"""具体命令:打开电灯"""
def __init__(self, light):
self.light = light
def execute(self):
self.light.turn_on()
class TurnOffCommand(Command):
"""具体命令:关闭电灯"""
def __init__(self, light):
self.light = light
def execute(self):
self.light.turn_off()
class RemoteControl:
"""调用者:遥控器,发出命令的请求"""
def __init__(self):
self.command = None
def set_command(self, command):
self.command = command
def press_button(self):
self.command.execute()
# 客户端代码
light = Light()
turn_on = TurnOnCommand(light)
turn_off = TurnOffCommand(light)
remote = RemoteControl()
remote.set_command(turn_on)
remote.press_button()
remote.set_command(turn_off)
remote.press_button()
一个文本编辑器的例子 使用命令模式来实现复制和粘贴功能:
class Command:
def execute(self):
pass
class TextEditor:
def __init__(self, text=''):
self.text = text
self.clipboard = ''
def copy(self):
self.clipboard = self.text
print(f"已复制: {self.clipboard}")
def paste(self):
self.text += self.clipboard
print(f"当前文本: {self.text}")
class CopyCommand(Command):
def __init__(self, editor):
self.editor = editor
def execute(self):
self.editor.copy()
class PasteCommand(Command):
def __init__(self, editor):
self.editor = editor
def execute(self):
self.editor.paste()
editor = TextEditor("Hello")
copy_cmd = CopyCommand(editor)
paste_cmd = PasteCommand(editor)
copy_cmd.execute()
paste_cmd.execute()
4.2 多级撤销功能
一个简单的计算器,能够执行操作并支持撤销功能:
class Command:
def execute(self):
pass
def undo(self):
pass
class Calculator:
def __init__(self):
self.value = 0
def add(self, num):
self.value += num
print(f"当前值: {self.value}")
def subtract(self, num):
self.value -= num
print(f"当前值: {self.value}")
class AddCommand(Command):
def __init__(self, calc, num):
self.calc = calc
self.num = num
def execute(self):
self.calc.add(self.num)
def undo(self):
self.calc.subtract(self.num)
calc = Calculator()
cmd = AddCommand(calc, 10)
cmd.execute()
cmd.undo()
4.3 事务型行为
模拟了数据库事务的基本操作:
class Command:
def execute(self):
pass
def rollback(self):
pass
class Database:
def __init__(self):
self.data = {}
def insert(self, key, value):
self.data[key] = value
print(f"插入: {key} -> {value}")
def delete(self, key):
if key in self.data:
del self.data[key]
print(f"删除: {key}")
class InsertCommand(Command):
def __init__(self, db, key, value):
self.db = db
self.key = key
self.value = value
def execute(self):
self.db.insert(self.key, self.value)
def rollback(self):
self.db.delete(self.key)
db = Database()
cmd = InsertCommand(db, 'id1', 'Data 1')
cmd.execute()
cmd.rollback()
4.4 排队请求
如何使用命令模式将请求排队并在不同的时间点处理:
from queue import Queue
class Command:
def execute(self):
pass
class Worker:
def do_something(self, msg):
print(f"工作内容: {msg}")
class DoSomethingCommand(Command):
def __init__(self, worker, msg):
self.worker = worker
self.msg = msg
def execute(self):
self.worker.do_something(self.msg)
queue = Queue()
worker = Worker()
queue.put(DoSomethingCommand(worker, "任务1"))
queue.put(DoSomethingCommand(worker, "任务2"))
while not queue.empty():
cmd = queue.get()
cmd.execute()
4.5 复合命令
通过命令模式实现的宏命令示例,可以执行一系列命令:
class Command:
def execute(self):
pass
class MacroCommand(Command):
def __init__(self):
self.commands = []
def add_command(self, cmd):
self.commands.append(cmd)
def execute(self):
for cmd in self.commands:
cmd.execute()
class SimpleCommand(Command):
def __init__(self, payload):
self.payload = payload
def execute(self):
print(f"执行: {self.payload}")
macro = MacroCommand()
macro.add_command(SimpleCommand("步骤1"))
macro.add_command(SimpleCommand("步骤2"))
macro.execute()相关推荐
- windows7安装版系统下载(win7安装版安装步骤)
-
1、win7系统装完后需要占用C盘空间10-15GB,如果把常用软件也安装到C盘,大小超过20+。 2、在分区的时候根据硬盘大小,如果硬盘相对较小,一般建议设置50G,最低不能低于30G。因...
- 韩剧网(韩剧网韩剧tv最新韩剧免费观看)
-
不收费,终身不收费~连续剧、综艺、歌曲,通通不要钱。之前好像是技术问题,现在恢复了。其它诸如人人视频、圈粉TV也都可以看韩剧哦。韩剧网有搜索页面的呀,搜索一下就可以了,不过好像《制作人》和玄彬的...
- 跳一跳游戏(跳一跳游戏是什么时候出来的)
-
GiveItUP《GiveItUP》是一款由Invictus推出的益智跑酷游戏。游戏与传统的跑酷类游戏最大的区别在于,游戏融入了高难度的节奏性挑战。红色火焰躲避技巧:红色火焰在游戏中是最常见的一个障碍...
- 什么是oa办公系统(办公oa系统的好处)
-
OA是OfficeAutomation的简写,就是办公自动化。所谓OA系统就是用网络和OA软件构建的一个单位内部的办公通信平台,用于辅助办公。OA系统完成单位内部的邮件通信、信息发布、文档管理、工作...
- 手机万能mp4转换器(手机万能mp4转换器下载)
-
1你需要先下载并安装一个叫格式工厂的软件。2然后打开->mp4。3点击添加文件的选项。4然后选择需要转换的视频,点击确定。5点击确定之后,再添加到转换列表中就可以。6再点...
- 迅雷beta官网(迅雷beta官方网)
-
若手机不能安装软件/游戏,建议:1.检查“未知来源”设置是否开启。2.查看下载的软件格式是否为机器所支持的。安卓系统手机支持的软件格式为.APK格式。3.下载软件时请阅读软件详情,查看是否对机器操作系...
- 私人家庭影院hd免费版游戏评测
-
《海上钢琴师》是由朱塞佩·托纳托雷执导,蒂姆·罗斯、比尔·努恩、梅兰尼·蒂埃里主演的剧情、音乐、爱情片。该片讲述了一个被命名为“1900”的弃婴在一艘远洋客轮上与钢琴结缘,成为钢琴大师的传奇故事。该片...
- 题库网(题库网站有哪些)
-
学科网的试卷有续费的和不需要交费的两类试卷,如果你是学科网的用户,可以下载试卷,还可以组卷,对于我们一线的老师来讲,这很方便的,只要学校肯出钱,在学科网前,全校的老师就可以注册免费试用学科,网上所有的...
- 类似迅雷的下载器有哪些(手机下载工具app)
-
推荐闪电下载和黑科技下载这两款APP手机上取代迅雷的下载器有闪电下载APP和黑科下载器APP等,以这两款为例,它们都是可以支持ed2k、磁力链等下载方式,也能够支持多种不同类似的文件的高速下载。、BT...
- qq号申请免费注册官网(qq注册账号免费申请网址)
-
你好,通用的申请QQ号码的方法如下:1、首先打开腾讯官网。2、在腾讯页面左边的通信工具里面点击号码。3、点击号码以后就会进入QQ注册页面。4、进入QQ注册页面以后,在昵称后面输入要注册号码的名字。5、...
- 安卓安装windows(安卓安装apk解析包出错)
-
oppo手机支持安卓和Windows操作系统,但是双系统安装需要手机支持虚拟化技术(VT)。如果手机支持VT,可以在电脑上安装虚拟化软件,然后在手机上运行VT兼容的虚拟化应用程序,从而实现在电脑上安装...
- 安全中心下载(网易安全中心下载)
-
可以通过解除保护模式页面,查看QQ帐号进入保护模式的原因,及获取恢复QQ帐号正常使用的方法。点击小i,然后把自动登录选项去掉,打开safari,随便点一个网页,去登录。畅游经常这样抽风,可以关了游戏...
- 重装系统步骤(USB重装系统步骤)
-
正版的win10及以上系统,现在已经都自带系统重置功能,可以一键重置恢复到出厂设置。1.按住win+i,在弹出的windows设置对话框中,点击更新和安全选项,在弹出的设置对话框中,点击左侧的恢复选项...
- qq群发软件手机版免费(qq群发软件安卓版免费下载)
-
分享一个可以免费试用的软件,“www.liaotiangou.com”在做微信营销的时候经常会需要使用到群发功能,出了多群群发之外,还有定时群发,甚至是定时发送朋友圈。多群群发可以解决以前最多只能群发...
- 小说排行榜2020前十名(耿美小说排行榜2020前十名)
-
1、元龙《元龙》是目前2020年哔哩哔哩国创动漫热门榜的第一名,讲述的是一名来自现代的王牌狙击手,穿越到了玄幻的元魂世界,从此展开的一系列热血的战斗与机遇,成为元魂世界的霸者的故事。2、灵笼《灵笼》是...
欢迎 你 发表评论:
- 一周热门
-
-
抖音上好看的小姐姐,Python给你都下载了
-
全网最简单易懂!495页Python漫画教程,高清PDF版免费下载
-
飞牛NAS部署TVGate Docker项目,实现内网一键转发、代理、jx
-
win7系统还原步骤图解(win7还原电脑系统的步骤)
-
Python 3.14 的 UUIDv6/v7/v8 上新,别再用 uuid4 () 啦!
-
python入门到脱坑 输入与输出—str()函数
-
16949认证费用是多少(16949审核员太难考了)
-
linux软件(linux软件图标)
-
Python三目运算基础与进阶_python三目运算符判断三个变量
-
(新版)Python 分布式爬虫与 JS 逆向进阶实战吾爱分享
-
- 最近发表
- 标签列表
-
- python计时 (73)
- python安装路径 (56)
- python类型转换 (93)
- python进度条 (67)
- python吧 (67)
- python的for循环 (65)
- python格式化字符串 (61)
- python静态方法 (57)
- python列表切片 (59)
- python面向对象编程 (60)
- python 代码加密 (65)
- python串口编程 (77)
- python封装 (57)
- python写入txt (66)
- python读取文件夹下所有文件 (59)
- python操作mysql数据库 (66)
- python获取列表的长度 (64)
- python接口 (63)
- python调用函数 (57)
- python多态 (60)
- python匿名函数 (59)
- python打印九九乘法表 (65)
- python赋值 (62)
- python异常 (69)
- python元祖 (57)
