百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术资源 > 正文

python设计模式之命令模式的组成、场景、特点、示例

off999 2024-10-27 11:57 38 浏览 0 评论

命令模式(Command Pattern)是一种常用的设计模式,属于行为型模式的一种。它的主要目的是将一个请求封装为一个对象,从而允许用户使用不同的请求、队列或者日志来参数化其他对象。命令模式也支持可撤销的操作。

1 命令模式的组成

命令模式通常包括以下几个角色:

  1. Command:命令接口,声明执行操作的接口。
  2. ConcreteCommand:具体命令,实现命令接口,并调用接收者的一个或多个动作来完成具体的命令。会保持一个接受者的引用。它是一个容器并不会执行具体处理
  3. Invoker:调用者,负责调用命令对象执行请求。
  4. Receiver:接收者,知道如何实施与执行一个请求相关的操作。
  5. Client:客户端,创建一个具体命令对象,并设置其接收者。

2 使用场景

命令模式在实际场景中的作用主要体现在以下几个方面:

  1. 解耦调用者和接收者
    命令模式可以将发出命令的对象和接收以及执行命令的对象解耦。这意味着命令的发送者不需要知道具体是谁执行的命令,也不需要知道命令是如何实现的。这种解耦使得系统更加灵活,易于扩展和修改。
  2. 命令的排队与执行
    命令模式允许将命令保存在队列中,然后在需要的时候依次执行。这在多线程程序中特别有用,可以在一个线程中排队命令,并在另一个线程中一一执行,有效地实现了线程间的任务调度。
  3. 支持撤销和重做操作
    命令模式可以通过实现撤销(undo)和重做(redo)命令来支持用户的撤销操作,这在编辑器、图形界面工具等需要提供用户操作回退功能的应用中非常有用。
  4. 支持操作的宏命令
    通过命令模式,可以将多个命令组合成一个宏命令,然后以单一命令的方式执行。这种方式常用于实现复杂的用户界面操作、事务性操作,或是需要一系列步骤共同完成一个任务的场景。
  5. 动态调整命令
    由于命令是对象,可以在运行时动态地改变命令的参数,或者完全更换执行的命令。这为动态调整程序的行为提供了极大的灵活性。
  6. 日志和事务
    命令模式可以用来记录详细的日志,记录用户的具体操作,或者在需要的时候,实现对命令的持久化。在软件崩溃后,这些记录可以用来恢复用户的操作。在事务型系统中,可以用命令模式来实现事务的提交和回滚。

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()

相关推荐

为啥系统重装后有两个系统(为啥系统重装后有两个系统 原来的系统还在)

电脑重装系统后有两个系统,需要重新安装,并且再安装系统时需要把原来的系统所在盘即C盘进行格式化,否则安装完成就还是两个系统,非常占系统内存。1、可能在安装时删除了原来的引导分区。2、可能安装时直接安装...

win10win7双系统引导设置(win10 win7双系统引导)

 步骤如下:  1、首先我是开机时按F8,进入安全模式界面。但是进去的时候等待了几分钟都无反应,上面一直显示请稍等。  2、没办法只有按下复位键重启电脑,因电脑之前装有一键GHOST备份,于是果断还原...

ie缓存清理在哪里(ie缓存如何清除)

?  1、首先打开IE浏览器,选择IE浏览器的工具这一选项;  2、下一步选择工具中的Internet的选项;  3、下一步就是在Internet选项中的常规的选项中;  4、选择常规--浏览历史记录...

华为正版鸿蒙40电脑操作系统下载中文版

安装华为鸿蒙40系统正式版需要先下载官方固件包,然后将固件包放到手机内部存储或外部存储卡中。打开手机设置,选择系统更新,点击“手动更新”,选择已下载的固件包进行安装。安装前请备份重要数据并确保手机电量...

笔记本电脑哪个牌子好用又实惠

1.神舟优雅X4优点:1.35kg很轻巧,14英寸够便携固态硬盘,速度快,有背光键盘。缺点:配置较低,只能轻度办公,售后一般。2.攀升MaxBookP1优点:零噪音,金属机身,固态硬盘,大触摸板,背...

电脑一开机就进入bios界面(电脑开机就会进入bios)

原因一:你的BIOS电池没有电了。解决方式:更换电池即可原因二:没有软驱但启用了软驱解决:可将软驱禁用——开机按DEL进BIOS,选择:STANDARDCMOSFEATURESDRIVEA:...

电脑windows7旗舰版怎么样(电脑windows7旗舰版好不好)

win7旗舰版挺好使的不过现在可以选择更win10。Windows7旗舰版属于微软公司开发的Windows7操作系统系统系列中的功能最高级的版本,也被叫做终结版本,是为了取代WindowsXP...

手机软件怎么升级最新版本呢
  • 手机软件怎么升级最新版本呢
  • 手机软件怎么升级最新版本呢
  • 手机软件怎么升级最新版本呢
  • 手机软件怎么升级最新版本呢
office2010老是弹出安装程序

没看到截图,最好是吧提示信息完整截图发上来。因为信息不会是仅仅“更改安装”几个字的。猜测是已经安装有Office2010了或原本的2010没有卸载干净。

win8玩游戏稳定吗(win8的游戏win10能玩吗)

1、确定驱动是最稳定的公版驱动,新驱动不一定适合游戏不要贸然升级。 2、确定电源已经设置为高性能模式。3、游戏过程开个游戏加加,可以自动为你切换独显,并且自动释放内存。也可以通过它注意下CPU占用,如...

win10系统更新版本(win10系统更新版本能回退吗)

win10怎么更新到1909版本win101909升级方法一:WindowsUpdate更新:1.依次点击开始—设置—更新和安全—windows更新—检查更新,需要更新补丁至最新,如果你经常不更新...

win7升级win10要留多少空间(windows7升级windows10需要多长时间)

win7电脑在系统已经激活并且开启系统更新的情况下,符合条件的系统会在右下角弹出windows10免费升级,直接点击确定就开始升级了。或者下载win10安助手,运行软件后会自动下载windows1...

国外比较开放的浏览器(国外比较开放的浏览器推荐)

1、打开控制面板。2、点击“检查防火墙状态”。3、点击左侧“高级设置”。4、选中“入栈规则”。5、右侧点击“新建规则”。6、选择“端口”。7、选择“TCP”,选中“特定端口”并输入你要开发的端口,或者...

一健ghost下载(一键ghost v2015.07.05)

你的是原版镜像,当然无法识别。。你可以使用微软usb工具。将镜像写入U盘或光盘。

纯净无毒的win7下载(有没有纯净的win7系统)

下面提供的是微软发布的Windows7各版本光盘ISO镜像下载地址,原始文件均来源自MSDN,和零-售彩盒版本光盘内容完全一致。请放心下载。(如果需要光盘的买家,请无视以下内容)下...

取消回复欢迎 发表评论: