Python命令模式介绍、使用

一、Python命令模式介绍

Python命令模式(Command Pattern)是一种行为型设计模式,它允许将请求或操作封装在对象中,并将其作为参数传递给调用对象,以在不同的环境中执行相同的请求或操作。

功能:

  • 将请求或操作与其接收者解耦,从而提高代码的灵活性和可重用性。
  • 支持撤销和重做操作,因为命令可以在历史记录中保存和管理。
  • 支持事务性操作,因为一组相关的命令可以组合成一个事务。

优点:

  • 命令模式可以使代码更加模块化和可扩展,因为命令可以轻松添加、删除或替换。
  • 命令模式使代码更加容易维护和测试,因为单个命令的代码只涉及一个操作,且命令可以在不同的环境中重用和测试。
  • 命令模式支持撤销操作,这增加了代码的可靠性和安全性。

缺点:

  • 命令模式的实现可能需要大量的代码,特别是当有许多不同的命令和接收者时。
  • 命令模式可能会导致代码的复杂性和间接性增加,因为它需要多个对象之间的交互。

应用场景:

  • 当需要将请求或操作封装在对象中并将其传递给调用对象时。
  • 当需要支持撤销和重做操作时。
  • 当需要支持事务性操作时。
  • 当需要动态添加、删除或替换命令时。

使用方式:

  • 创建一个命令接口,它包含执行和撤销方法。
  • 创建一个或多个命令类,它们实现命令接口中的方法,并包含命令相关的数据和操作。
  • 创建一个调用对象,它接收命令对象并将其存储在一些数据结构中,以便在需要时执行、撤销或重做它们。
  • 创建一个接收者对象,它实现命令类中的操作。
  • 命令类将接收者对象与操作相关联,并在执行或撤销时调用其方法。

在应用程序开发中的使用:

  • 撤销和重做操作:可以使用命令模式将受影响的操作保存在历史记录中,并支持撤销和重做操作。
  • 动态调用:可以使用命令模式将请求和调用对象解耦,从而允许动态配置调用对象。
  • 事务性操作:可以使用命令模式将一组相关的操作组合成一个事务,从而确保它们都成功或都失败。
  • 日志记录:可以使用命令模式将操作和结果记录在日志文件中,以便跟踪和调试应用程序。

二、命令模式使用

工作原理:

  • 调用对象接收命令对象,并将其存储在一个数据结构中,以便在需要时执行、撤销或重做它们。
  • 命令对象实现命令接口中的执行和撤销方法,并包含命令相关的数据和操作。
  • 调用对象将命令对象传递给接收者对象,它实现命令类中的操作。
  • 在执行时,命令对象调用接收者对象的方法以执行操作。在撤销时,命令对象调用接收者对象的方法以撤销操作。
  • 可以使用历史记录来存储命令对象,支持撤销和重做操作。

示例一:实现动态调用功能

在实现动态调用功能时,命令模式可以帮助我们实现很好的可扩展性和解耦。以下是一个简单的示例,其中我们使用命令模式来实现动态调用功能。

首先,我们定义一个命令接口,所有的命令都需要实现这个接口。

接下来,我们实现具体的命令,如下所示:

然后,我们需要有一个接收者来执行具体的操作。在这个例子中,我们不需要接收者。

接下来,我们定义一个 Invoker 类,它接收命令并在需要的时候调用它们。在这个例子中,我们使用 Invoker 来动态调用命令。

现在,我们可以使用 Invoker 来执行命令。

from abc import  ABC, abstractmethod# 定义抽象类:命令接口
class Command(ABC):@abstractmethoddef execute(self):pass# 定义实现类:实现具体命令
class test1Command(Command):def execute(self):print("test1 command")class test2Command(Command):def execute(self):print("test2 command")# 定义调用程序类
class Invoker:def __init__(self):self.commands = {}def set_command(self, name, command):self.commands[name] = command # 存入字典def execute_command(self, name):if name in self.commands:self.commands[name].execute()else:print(f"{name} not found!")# 创建实例
invoker = Invoker()invoker.set_command("test1", test1Command())
invoker.execute_command("test1")invoker.set_command("test2", test2Command())
invoker.execute_command("test2")invoker.execute_command("test")

运行结果:

test1 command
test2 command
test not found!

注意,在这个例子中,我们使用了 Invoker 来管理命令,并且使用命令的名称来动态调用命令。如果我们尝试执行没有添加到 Invoker 中的命令,那么我们就会收到一个错误消息。此外,我们可以使用 Invoker 来添加和删除命令,从而实现更好的可扩展性。

示例二:实现撤销和重做操作

在实现撤销和重做操作的场景中,命令模式可以实现很好的解耦和可扩展性。以下是一个简单的示例,其中我们使用命令模式来实现撤销和重做操作。

首先,我们定义一个命令接口,所有的命令都需要实现这个接口。

接下来,我们实现具体的命令,如下所示:

然后,我们需要有一个接收者来执行具体的操作。在这个例子中,我们定义了一个简单的列表类来实现添加和删除操作。

接下来,我们定义一个 Invoker 类,它接收命令并在需要的时候调用它们。

现在,我们可以使用 Invoker 来执行命令。

from abc import  ABC, abstractmethod# 定义抽象类:命令接口
class Command(ABC):@abstractmethoddef execute(self):pass@abstractmethoddef undo(self):pass@abstractmethoddef redo(self):pass# 定义实现类:实现具体命令
class AddCommand(Command):def __init__(self, receiver, value):self.receiver = receiverself.value = valuedef execute(self):self.receiver.add(self.value)def undo(self):self.receiver.remove(self.value)def redo(self):self.execute()class RemoveCommand(Command):def __init__(self, receiver, value):self.receiver = receiverself.value = valuedef execute(self):self.receiver.remove(self.value)def undo(self):self.receiver.add(self.value)def redo(self):self.execute()# 定义接收者
class Receiver():def __init__(self):self.data = []def add(self, item):self.data.append(item)print(f"Add item: {item}")def remove(self, item):self.data.remove(item)print(f"Remove item: {item}")def show(self):print("Current data: ", self.data)# 定义调用程序类
class Invoker:def __init__(self):self.commands = []self.index = -1def set_command(self,command):self.commands.append(command) # 存入数组self.index += 1def execute_command(self):self.commands[self.index].execute()def undo_command(self):if self.index >= 0:self.commands[self.index].undo()self.index -= 1def redo_command(self):if self.index < len(self.commands) -1:self.index += 1self.commands[self.index].redo()# 创建实例
receiver = Receiver()
invoker = Invoker()
print("1----add")
add_command = AddCommand(receiver, 1)
invoker.set_command(add_command)
invoker.execute_command()
receiver.show()
print("2----add")
add_command = AddCommand(receiver, 2)
invoker.set_command(add_command)
invoker.execute_command()
receiver.show()
print("3----remove")
remove_command = RemoveCommand(receiver, 1)
invoker.set_command(remove_command)
invoker.execute_command()
receiver.show()
print("4----undo")
invoker.undo_command()
receiver.show()
print("5----redo")
invoker.redo_command()
receiver.show()

运行结果:

1----add
Add item: 1
Current data:  [1]
2----add
Add item: 2
Current data:  [1, 2]
3----remove
Remove item: 1
Current data:  [2]
4----undo
Add item: 1
Current data:  [2, 1]
5----redo
Remove item: 1
Current data:  [2]

注意,在这个例子中,我们使用了 Invoker 来管理命令,并且每次调用 execute_command 方法时,都会执行当前命令。当我们需要执行撤销操作时,我们会调用 undo_command 方法,并将当前命令的索引减去 1。当我们需要执行重做操作时,我们会调用 redo_command 方法,并将当前命令的索引加上 1。如果我们已经达到了命令队列的末尾或开头,那么我们就不会执行任何操作。

实例三:实现日志记录功能

在实现日志记录功能时,命令模式可以帮助我们记录每个操作的详细信息,包括操作执行的时间、执行者、执行的结果等。以下是一个简单的示例,其中我们使用命令模式来实现日志记录功能。

首先,我们定义一个命令接口,所有的命令都需要实现这个接口。

每个命令需要实现 execute 方法和 undo 方法。execute 方法实际执行命令逻辑,undo 方法则实现命令的撤销操作。

接下来,我们实现具体的命令,定义了一个具体的命令AddCommand。这个命令需要一个接收者(receiver),它是负责实际执行操作的对象。在这个例子中,使用一个Receiver类来表示接收者。命令模式还提供了一个 log 方法,用于记录每个操作的详细信息。

使用 Invoker 来执行命令和撤销命令。Invoker可以维护一个命令队列,并同时执行多个命令。

在这个例子中,我们使用了 Invoker 来管理命令队列,并在需要时依次执行它们。我们还实现了一个 undo_commands 方法来撤销最后一个命令,并且在需要时打印当前状态。

from abc import ABC, abstractmethod# 定义逻辑类
class Command(ABC):@abstractmethoddef execute(self): # 实现执行命令逻辑pass@abstractmethoddef undo(self):   # 实现命令撤销pass# 定义具体实现类
class AddComand(Command):def __init__(self, receiver, value):self.receiver = receiver # 接收者self.value = valuedef execute(self):result = self.receiver.add(self.value) # 调用Receiver类的add()方法self.log(result)def undo(self):result = self.receiver.remove(self.value) # 调用Receiver类的remove()方法self.log(result)def log(self, result):print(f"added {self.value} to receiver {self.receiver} with result {result}")class RemoveComand(Command):def __init__(self, receiver, value):self.receiver = receiver # 接收者self.value = valuedef execute(self):result = self.receiver.remove(self.value) # 调用Receiver类的add()方法self.log(result)def undo(self):result = self.receiver.add(self.value) # 调用Receiver类的remove()方法self.log(result)def log(self, result):print(f"Remove {self.value} to receiver {self.receiver} with result {result}")# 定义接收者
class Receiver():def __init__(self):self.items = []def add(self, item):self.items.append(item)return Truedef remove(self, item):if item in self.items:self.items.remove(item)return Truereturn Falsedef __str__(self): # 定义对象的字符串表示形式return str(self.items)# 定义Invoker(调用程序)维护命令队列:执行命令、撤销命令
class Invoker():def __init__(self):self.commands = []self.undo_commands = []def set_command(self, command):self.commands.append(command)def execute_command(self):for command in self.commands:command.execute() # 调用具体实现类:执行方法execute()self.undo_commands.append(command)self.commands = []def undo_command(self):num_commands = len(self.undo_commands)if num_commands == 0:returncommand = self.undo_commands.pop() # pop()方法删除列表末尾元素,返回列表末尾元素。 pop(0)删除列表第一个元素。command.undo()   # 调用具体实现类:撤销方法undo()def show(self):for command in self.undo_commands:command.show()# 创建实例
receiver = Receiver()
invoke = Invoker()add_command = AddComand(receiver, 1)
invoke.set_command(add_command)
# invoke.execute_command()add_command = AddComand(receiver, 2)
invoke.set_command(add_command)
# invoke.execute_command()add_command = AddComand(receiver, 3)
invoke.set_command(add_command)invoke.execute_command()remove_command = RemoveComand(receiver, 2)
invoke.set_command(remove_command)
invoke.execute_command()print(receiver)

运行结果:

added 1 to receiver [1] with result True
added 2 to receiver [1, 2] with result True
added 3 to receiver [1, 2, 3] with result True
Remove 2 to receiver [1, 3] with result True

在这个例子中,我们首先添加三个项目到接收者中,然后删除一个项目。我们看到了添加操作的详细信息,包括操作执行的时间、执行者、执行的结果等。

在实际应用中,我们可以将每个命令的详细信息写入日志文件中,以便后续跟踪和调试。同时,我们还可以将日志信息发送到远程服务器上进行中央日志管理。这样可以帮助我们更好地管理和维护系统。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/14373.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

SqlServer读写分离对等发布

SqlServer读写分离对等发布: 对等发布支持多主复制。发布服务器将事务流式传输到拓扑中的所有对等方。所有对等节点可以读取和写入更改,且所有更改将传播到拓扑中的所有节点。 注意点: 1.各服务器的数据库名字要保证一样。 2.发布名称必须保持一致。 3.各服务器必须都是…

【论文阅读】The Deep Learning Compiler: A Comprehensive Survey

论文来源&#xff1a;Li M , Liu Y , Liu X ,et al.The Deep Learning Compiler: A Comprehensive Survey[J]. 2020.DOI:10.1109/TPDS.2020.3030548. 这是一篇关于深度学习编译器的综述类文章。 什么是深度学习编译器 深度学习&#xff08;Deep Learning&#xff09;编译器将…

【Maven】让maven更高效,优化maven构建项目速度

打开idea的setting&#xff0c;找到maven&#xff0c;设置它多线程数&#xff0c;重启后即可&#xff01; 我这里是8&#xff0c;你们可以随便设置。 如下图&#xff1a;

前端请求传输token到后端的两种方式

谈谈 cookie & session & jwt - 掘金 前端在每次请求时把token放在请求头中发送给后端&#xff0c;目前有两种方式&#xff1a; 一是通过cookie的形式&#xff0c;即把token放在cookie中&#xff0c;每次浏览器会自动帮我们带过去&#xff0c;不需要我们自己设置。 二…

flutter实战(01)windows桌面版 修改应用logo、名称、显示位置、显示大小

说明&#xff1a;该系列文章主要为flutter在windows桌面平台实战中遇到的一些坑。 1 修改logo 只需要在flutter项目/windows/runner/resources目录下替换原来的应用图标 app_icon.ico即可。 2 修改应用名称、显示位置、显示大小 修改flutter项目/windows/runner/main.cpp 文…

Github git clone 和 git push 特别慢的解决办法

1.在本地上使用 SSH 命令无法git push 上传 github 项目 2.使用 git clone 下载项目特别慢总是加载不了 解决办法1 将 *** 的连接模式换成&#xff1a;D-i-r-e-c-t&#xff08;好像不太有用&#xff09; 后面再找找能不能再G-l-o-b-a-l 下解决该问题 解决办法 2 mac下直接设…

自定义AOP记录请求日志

Aspect Component public class LogAspect {public LogAspect() {System.out.println("LogAspect");}private final static Logger LOG LoggerFactory.getLogger(LogAspect.class);/*** 定义一个切点 TODO 记得替换为自己项目的controller位置*/Pointcut("exec…

Spring Data【Spring Data Redis、Spring Data ElasticSearch】(二)-全面详解(学习总结---从入门到深化)

目录 四、 Spring Data Redis 五、 Spring Data ElasticSearch 四、 Spring Data Redis Redis 是一个基于内存的数据结构存储系统&#xff0c;它可以用作数据库或者缓存。它支持多种 类型的数据结构&#xff0c;这些数据结构类型分别为 String&#xff08;字符串&#xff09…

lab3 attacklab

phase1 首先&#xff0c;通过这个指令生成前三个阶段要用到的ctarget的反汇编代码objdump -d ./ctarget >> target.s 其次&#xff0c;假设我们的输入文件是touch1.txt&#xff0c;那么运行这个指令就可以去验证是否是正确的./hex2raw <touch1.txt | ./ctarget -q 整…

MethodInterceptor

目录 1 MethodInterceptor 1.1 /// This will be called via Reflection 1.2 HandleAsync MethodInterceptor /// This will be called via Reflection /// </summary> /// <typeparam name"TResult"></typeparam> /// <param name"…

Flutter 开发者工具 Android Studio 开发Flutter应用

Flutter 开发者工具 在 Android Studio 开发Flutter应用 &#x1f525; Android Studio 版本更新 &#x1f525; Android Studio Check for Update Connection failed ​ 解决方案 如果是运行的是32位的android studio需要在andriod studio的启动目录下找到studio.exe.vmoptio…

WIZnet W5100S-EVB-Pico DHCP 配置教程(三)

DHCP协议介绍 什么是DHCP&#xff1f; 动态主机配置协议DHCP&#xff08;Dynamic Host Configuration Protocol&#xff09;是一种网络管理协议&#xff0c;用于集中对用户IP地址进行动态管理和配置。 DHCP于1993年10月成为标准协议&#xff0c;其前身是BOOTP协议。DHCP协议由…

Python批量下载主播照片,实现人脸识别, 进行颜值评分,制作颜值排行榜

昨晚一回家&#xff0c;表弟就神神秘秘的跟我说&#xff0c;发现一个高颜值网站&#xff0c;非要拉着我研究一下她们的颜值高低。 我心想&#xff0c;这还得要我一个个慢慢看&#xff0c;太麻烦了~ 于是反手用Python给他写了一个人脸识别代码&#xff0c;把她们的照片全部爬下…

【华为OD机试】 选修课

题目描述 现有两门选修课&#xff0c;每门选修课都有一部分学生选修&#xff0c;每个学生都有选修课的成绩&#xff0c;需要你找出同时选修了两门选修课的学生&#xff0c;先按照班级进行划分&#xff0c;班级编号小的先输出&#xff0c;每个班级按照两门选修课成绩和的降序排序…

【TiDB理论知识 07】SQL执行流程

一 DML语句读写流程 1 DML语句读流程概要 用户发出SQL 被协议层接收 Protocal Layer 通过PD获取时间戳 parse模块 解析SQL&#xff0c;通过词法解析 与 语法解析 生成AST语法树 编译SQL Compile模块 ,区分点查 与 非点查&#xff0c;生成执行计划 发送给Executor,从TIKV获…

AAOS 音频焦点请求

文章目录 前言基本概念提供给应用来获取音频焦点的apiAAOS中的音频焦点管理交互矩阵duck的实现流程AAOS 测试应用kitchensink焦点相关 前言 本文章的目标是首先了解Android中音频焦点的基本概念&#xff0c;理解代码中相关音频焦点的使用方法。其次理解AAOS 中相关交互矩阵概念…

[SQL挖掘机] - 字符串函数 - substring

介绍: substring函数是在mysql中用于提取字符串的一种函数。它接受一个字符串作为输入&#xff0c;并返回从该字符串中指定位置开始的一部分子串。substring函数可以用于获取字符串中的特定字符或子串&#xff0c;以便进行进一步的处理或分析。 用法: 下面是substring函数的…

【Linux】Centos的一些快捷操作

Centos的一些快捷操作 一个窗口多个终端GVIM 一个窗口多个文件 一个窗口多个终端 GVIM 一个窗口多个文件

Hadoop学习指南:探索大数据时代的重要组成——Hadoop概述

前言 在当今大数据时代&#xff0c;处理海量数据成为了一项关键任务。Hadoop作为一种开源的分布式计算框架&#xff0c;为大规模数据处理和存储提供了强大的解决方案。本文将介绍Hadoop的组成和其在大数据处理中的重要作用&#xff0c;让我们一同踏上学习Hadoop的旅程。 Hado…

Jenkins集成SonarQube保姆级教程

Jenkins是自动化部署平台&#xff0c;一个粗眉大眼的糙汉子&#xff01; SonarQube是代码扫描平台&#xff0c;一个眉目清秀的小女子&#xff01; 有一天&#xff0c;上天交给我一个任务&#xff0c;去撮合撮合他们&#xff01; 我抬头看了看天&#xff0c; 不&#xff0c;…