python设计模式之命令模式的组成、场景、特点、示例
off999 2024-10-27 11:57 30 浏览 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()相关推荐
- 阿里云国际站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)是一个开源的容器编排系统,用于自动化部署、扩展和管理容器化应用程序。它...
欢迎 你 发表评论:
- 一周热门
- 最近发表
- 标签列表
-
- 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)
