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

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

off999 2024-10-27 11:57 30 浏览 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()

相关推荐

阿里云国际站ECS:阿里云ECS如何提高网站的访问速度?

TG:@yunlaoda360引言:速度即体验,速度即业务在当今数字化的世界中,网站的访问速度已成为决定用户体验、用户留存乃至业务转化率的关键因素。页面加载每延迟一秒,都可能导致用户流失和收入损失。对...

高流量大并发Linux TCP性能调优_linux 高并发网络编程

其实主要是手里面的跑openvpn服务器。因为并没有明文禁p2p(哎……想想那么多流量好像不跑点p2p也跑不完),所以造成有的时候如果有比较多人跑BT的话,会造成VPN速度急剧下降。本文所面对的情况为...

性能测试100集(12)性能指标资源使用率

在性能测试中,资源使用率是评估系统硬件效率的关键指标,主要包括以下四类:#性能测试##性能压测策略##软件测试#1.CPU使用率定义:CPU处理任务的时间占比,计算公式为1-空闲时间/总...

Linux 服务器常见的性能调优_linux高性能服务端编程

一、Linux服务器性能调优第一步——先搞懂“看什么”很多人刚接触Linux性能调优时,总想着直接改配置,其实第一步该是“看清楚问题”。就像医生看病要先听诊,调优前得先知道服务器“哪里...

Nginx性能优化实战:手把手教你提升10倍性能!

关注△mikechen△,十余年BAT架构经验倾囊相授!Nginx是大型架构而核心,下面我重点详解Nginx性能@mikechen文章来源:mikechen.cc1.worker_processe...

高并发场景下,Spring Cloud Gateway如何抗住百万QPS?

关注△mikechen△,十余年BAT架构经验倾囊相授!大家好,我是mikechen。高并发场景下网关作为流量的入口非常重要,下面我重点详解SpringCloudGateway如何抗住百万性能@m...

Kubernetes 高并发处理实战(可落地案例 + 源码)

目标场景:对外提供HTTPAPI的微服务在短时间内收到大量请求(例如每秒数千至数万RPS),要求系统可弹性扩容、限流降级、缓存减压、稳定运行并能自动恢复。总体思路(多层防护):边缘层:云LB...

高并发场景下,Nginx如何扛住千万级请求?

Nginx是大型架构的必备中间件,下面我重点详解Nginx如何实现高并发@mikechen文章来源:mikechen.cc事件驱动模型Nginx采用事件驱动模型,这是Nginx高并发性能的基石。传统...

Spring Boot+Vue全栈开发实战,中文版高清PDF资源

SpringBoot+Vue全栈开发实战,中文高清PDF资源,需要的可以私我:)SpringBoot致力于简化开发配置并为企业级开发提供一系列非业务性功能,而Vue则采用数据驱动视图的方式将程序...

Docker-基础操作_docker基础实战教程二

一、镜像1、从仓库获取镜像搜索镜像:dockersearchimage_name搜索结果过滤:是否官方:dockersearch--filter="is-offical=true...

你有空吗?跟我一起搭个服务器好不好?

来人人都是产品经理【起点学院】,BAT实战派产品总监手把手系统带你学产品、学运营。昨天闲的没事的时候,随手翻了翻写过的文章,发现一个很严重的问题。就是大多数时间我都在滔滔不绝的讲理论,却很少有涉及动手...

部署你自己的 SaaS_saas如何部署

部署你自己的VPNOpenVPN——功能齐全的开源VPN解决方案。(DigitalOcean教程)dockovpn.io—无状态OpenVPNdockerized服务器,不需要持久存储。...

Docker Compose_dockercompose安装

DockerCompose概述DockerCompose是一个用来定义和管理多容器应用的工具,通过一个docker-compose.yml文件,用YAML格式描述服务、网络、卷等内容,...

京东T7架构师推出的电子版SpringBoot,从构建小系统到架构大系统

前言:Java的各种开发框架发展了很多年,影响了一代又一代的程序员,现在无论是程序员,还是架构师,使用这些开发框架都面临着两方面的挑战。一方面是要快速开发出系统,这就要求使用的开发框架尽量简单,无论...

Kubernetes (k8s) 入门学习指南_k8s kubeproxy

Kubernetes(k8s)入门学习指南一、什么是Kubernetes?为什么需要它?Kubernetes(k8s)是一个开源的容器编排系统,用于自动化部署、扩展和管理容器化应用程序。它...

取消回复欢迎 发表评论: