当前位置: 首页 > news >正文

命令模式 (Command Pattern)

定义

命令模式(Command Pattern)是一种行为型设计模式,它将一个请求封装为一个对象,从而允许用户使用不同的请求、队列或日志来参数化其他对象。命令模式也支持可撤销的操作。主要目的是将命令的发送者和接收者解耦,引入了命令对象来在发送者和接收者之间充当中介。

命令模式主要涉及以下角色:

  • 命令(Command)接口:声明执行操作的接口。
  • 具体命令(Concrete Command):实现命令接口,定义了接收者和行为之间的绑定关系。调用者通过执行命令来执行相应的操作。
  • 接收者(Receiver):知道如何实施与执行一个请求相关的操作。任何类都可能作为一个接收者。
  • 调用者(Invoker):要求该命令执行这个请求。通常会持有命令对象,并在某个时间点调用命令对象的执行方法。
  • 客户端(Client):创建具体命令对象并设置其接收者。
解决的问题
  • 命令的封装和抽象
    • 当需要将请求或简单操作封装为一个对象时,命令模式提供了一种方式。这样,请求的发送者和接收者可以被解耦,使得发送者不需要知道请求的具体实现细节。
  • 可变更和扩展命令
    • 命令模式允许动态地更改、添加或删除命令,而无需修改现有代码。这增加了系统的灵活性,并支持动态的命令更改。
  • 历史记录和撤销功能
    • 命令模式可以记录所有执行的操作和命令,这使得实现撤销(undo)和重做(redo)功能变得可能。
  • 队列请求和日志请求
    • 命令对象可以被序列化和存储,以便稍后执行或远程执行,支持队列请求和日志请求的功能。
  • 参数化对象
    • 通过命令模式,可以参数化对象,根据不同的命令改变对象的行为。
  • 操作的解耦
    • 命令模式将发起操作的对象和执行操作的对象解耦,提高了系统的模块化,使得命令的发送者和接收者不直接交互。
使用场景
  • 参数化对象
    • 当需要基于不同的命令改变对象的行为时,命令模式可以将这些命令封装成对象,实现对象行为的动态切换。
  • 操作的队列化和日志记录
    • 在需要对命令进行排队、记录或执行延迟时,命令模式非常有用。例如,在实现一个多线程的任务队列、事务队列或日志系统时,可以使用命令模式来封装具体操作。
  • 撤销和重做功能
    • 在应用程序(如编辑器)中实现撤销(undo)和重做(redo)功能时,命令模式可以记录每一步操作并提供回滚机制。
  • 分离命令的发起者和执行者
    • 当需要将命令的发起者和执行者解耦时,命令模式提供了一种机制,让发起者不需要了解命令的具体执行过程。
  • 组合命令
    • 在需要组合多个命令以形成复合命令时,命令模式使得可以将多个简单命令组合成一个宏命令。
  • 回调功能的实现
    • 在需要提供回调功能时,命令模式允许将操作封装成对象,并在合适的时候执行。
示例代码1-简单的遥控器
// 命令接口
interface Command {void execute();
}// 具体命令:开灯
class LightOnCommand implements Command {private Light light;public LightOnCommand(Light light) {this.light = light;}public void execute() {light.turnOn();}
}// 具体命令:关灯
class LightOffCommand implements Command {private Light light;public LightOffCommand(Light light) {this.light = light;}public void execute() {light.turnOff();}
}// 请求者类:遥控器
class RemoteControl {private Command command;public void setCommand(Command command) {this.command = command;}public void pressButton() {command.execute();}
}// 接收者类:灯
class Light {public void turnOn() {System.out.println("Light is on.");}public void turnOff() {System.out.println("Light is off.");}
}// 客户端代码
public class CommandDemo {public static void main(String[] args) {Light light = new Light();Command lightOn = new LightOnCommand(light);Command lightOff = new LightOffCommand(light);RemoteControl remote = new RemoteControl();remote.setCommand(lightOn);remote.pressButton();  // 输出: Light is on.remote.setCommand(lightOff);remote.pressButton();  // 输出: Light is off.}
}
示例代码2-文本编辑器的命令
// 命令接口
interface Command {void execute();void undo();
}// 具体命令:写入文本
class WriteCommand implements Command {private TextFile textFile;private String textToWrite;public WriteCommand(TextFile textFile, String text) {this.textFile = textFile;this.textToWrite = text;}public void execute() {textFile.write(textToWrite);}public void undo() {textFile.eraseLast();}
}// 接收者类:文本文件
class TextFile {private StringBuilder text = new StringBuilder();public void write(String text) {this.text.append(text);}public void eraseLast() {if (text.length() > 0) {text.deleteCharAt(text.length() - 1);}}public String read() {return text.toString();}
}// 客户端代码
public class EditorDemo {public static void main(String[] args) {TextFile textFile = new TextFile();Command writeHello = new WriteCommand(textFile, "Hello");Command writeWorld = new WriteCommand(textFile, " World");writeHello.execute();writeWorld.execute();System.out.println(textFile.read()); // 输出: Hello WorldwriteWorld.undo();System.out.println(textFile.read()); // 输出: Hello}
}
主要符合的设计原则
  • 开闭原则(Open-Closed Principle)
    • 命令模式允许在不修改已有代码的情况下引入新的命令。你可以添加新的命令类来扩展系统的行为,而无需修改现有的请求发送者或其他命令的代码。因此,系统对扩展是开放的,但对修改是封闭的。
  • 单一职责原则(Single Responsibility Principle)
    • 在命令模式中,每个命令类只负责一个特定的操作或行为,这符合单一职责原则。此外,请求的发送者和接收者也被解耦,每个对象都只处理它自己的部分。
  • 依赖倒置原则(Dependency Inversion Principle)
    • 命令模式中,请求发送者依赖于命令抽象,而不是具体的命令实现。这符合依赖倒置原则,即高层模块不应该依赖于低层模块,两者都应该依赖于抽象。
在JDK中的应用
  • java.lang.Runnable
    • Runnable 接口是命令模式的一个经典例子。它代表一个待执行的操作,通常用于创建线程。实现 Runnable 接口的类封装了要在线程中执行的操作(命令),而 Thread 类则充当调用者,执行这些操作。
  • javax.swing.Action
    • 在Swing库中,Action 接口用于封装事件监听器的响应逻辑。当触发事件(如按钮点击)时,与之关联的 Action 对象被执行,这类似于命令模式中命令的执行。
  • java.util.concurrent.Callable
    • 类似于 RunnableCallable 接口也封装了一个异步操作,但与 Runnable 不同的是,Callable 可以返回一个结果并能抛出异常。Callable 实例可以提交给 ExecutorService 执行,后者充当调用者的角色。

这些例子虽然不是命令模式的完整实现,但它们体现了命令模式的核心概念:封装操作作为一个对象,并由另一个对象负责执行这些操作。这种模式在JDK中的应用提供了一种灵活的方式来处理异步操作、事件处理和其他需要动态指定和执行操作的场景。

在Spring中的应用
  • Spring的事务管理
    • 在Spring的事务管理中,事务代码可以被看作是命令对象。当执行事务时,实际的事务逻辑(即命令)被封装在事务管理器中。这样的设计允许灵活地启动、提交或回滚事务,类似于命令模式中的命令执行。
  • Spring MVC的Controller方法
    • 在Spring MVC中,控制器(Controller)中的方法可以被视为命令。每个方法封装了特定的处理逻辑,而DispatcherServlet则相当于命令的调用者,根据请求调用相应的控制器方法。
  • JdbcTemplate的回调
    • Spring的 JdbcTemplate 类使用回调机制来封装数据库操作,这些回调可以被视为命令对象。JdbcTemplate 提供了一系列的方法,比如 queryupdate,这些方法接受一个回调参数,封装了执行操作的具体逻辑。

虽然这些例子并不是命令模式的纯粹形式,但它们在设计上采用了命令模式的核心思想——将请求封装成对象,并由另一个对象来执行这些请求。这种模式在Spring框架中主要体现在将业务逻辑封装在独立的类或方法中,而由框架的其他部分来负责调用这些逻辑。这样的设计有助于降低系统各部分之间的耦合度,提高代码的可重用性和灵活性。


相关文章:

命令模式 (Command Pattern)

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

蓝桥杯官网练习题(奇怪的数列)

题目描述 从 X 星截获一份电码,是一些数字,如下: 13 1113 3113132113 1113122113 ⋯ YY 博士经彻夜研究,发现了规律: 第一行的数字随便是什么,以后每一行都是对上一行"读出来" 比如第 2…...

flink的异常concurrent.TimeoutException: Heartbeat of TaskManager with id的解决

背景 在使用flink进行集成测试时,我们会使用MiniClusterWithClientResource类,但是当我们断点导致在某个方法执行的时间比较长时,会有错误发生,那么该如何解决这个错误呢? 处理concurrent.TimeoutException: Heartbe…...

火电安全事故vr模拟仿真培训强交互更真实

VR消防,利用VR虚拟现实技术,将VR和消防教育融合在一起达到寓教于乐的效果, VR消防教育是对于家中、校园内、大型商场、公司办公室等情景产品研发的消防安全培训类VR系统软件,根据互动体验、互动、视角实际操作、视听觉系统多度自然…...

ELK企业级日志分析平台

目录 一、elasticsearch 1、集群部署 2、cerebro部署 3、elasticsearch-head插件部署 4、elasticsearch集群角色分类 二、logstash 1、部署 2、elasticsearch输出插件 3、file输入插件 4、file输出插件 5、syslog 插件 6、多行过滤插件 7、grok过滤 三、kibana数…...

.NET面试题1

1.什么是C#? C#(读作"C sharp")是一种通用的、面向对象的编程语言,由Microsoft开发。它是一种静态类型语言,支持强类型检查和面向对象编程(OOP)的概念。C#主要用于开发Windows应用程序…...

mongodb 日志详情

1 mongodb日志简介 MongoDB的日志包括两个主要部分:操作日志(oplog)和系统日志。 1.1 操作日志 操作日志(oplog)是一个特殊的集合,用于记录所有对数据库进行的操作(如插入、更新和删除&#x…...

Oracle中文显示???????解决办法

项目场景: Oracleoracle中文显示???解决办法 问题描述 原因分析: Oracle中文显示???通常是由于字符集不匹配或者编码问题导致的。当数据库中的数据使用的是某种字符集,而客户端或者应用程序使用的是另一种字符集时,就会出…...

Java查询数据放入word模板中并在前端导出下载

需求:查询数据放入word模板中并在前端导出下载 解决方法:在模板的位置定义参数如 {{name}} {{age}}等等,使用 poi 处理 伪代码: PostMapping("/practiceAppr")public AjaxResult practiceAppr(OutputStream outputSt…...

HarmonyOS ArkTS 应用添加弹窗(八)

概述 在我们日常使用应用的时候,可能会进行一些敏感的操作,比如删除联系人,这时候我们给应用添加弹窗来提示用户是否需要执行该操作,如下图所示: 弹窗是一种模态窗口,通常用来展示用户当前需要的或用户必须…...

排序算法-----快速排序(非递归实现)

目录 前言 快速排序 基本思路 非递归代码实现 算法分析 空间复杂度 时间复杂度 稳定性 前言 很久没跟新数据结构与算法这一栏了,因为数据结构与算法基本上都发布完了,哈哈,那今天我就把前面排序算法那一块的快速排序完善一下&#xff0…...

el-input限制输入整数等分析

文章目录 前言1、在 Vue 中,可以使用以下几种方式来限制 el-input 只能输入整数1.1 设置input 的 type为number1.2 使用inputmode1.3 使用自定义指令1.4 使用计算属性1.5 使用 onafterpaste ,onkeyup1.6 el-input-number 的precision属性 总结 前言 input 限制输入…...

医院手术麻醉信息系统全套源码,自主版权,支持二次开发

医院手术麻醉信息系统全套商业源码,自主版权,支持二次开发 手术麻醉信息系统是HIS产品的中的一个组成部分,主要应用于医院的麻醉科,属于电子病历类产品。医院麻醉监护的功能覆盖整个手术与麻醉的全过程,包括手术申请与…...

canvas扩展001:利用fabric绘制图形,可以平移,旋转,放缩

canvas可以使用Fabric.js来做扩展,您可以在画布上创建和填充对象; 诸如简单几何形状之类的对象 - 矩形、圆形、椭圆形、多边形或由数百或数千条简单路径组成的更复杂的形状。 然后,您可以使用鼠标缩放、移动和旋转这些对象; 修改它…...

什么是机器学习

前言 机器学习(Machine Learning, ML)是一个总称,用于解决由各位程序员自己基于 if-else 等规则开发算法而导致成本过高的问题,想要通过帮助机器 「发现」 它们 「自己」 解决问题的算法来解决 ,而不需要程序员将所有…...

电子桌牌如何赋能数字化会务?以深圳程序员节为例。

10月24日,由深圳市人民政府指导,深圳市工业和信息化局、龙华区人民政府、国家工业信息安全发展研究中心、中国软件行业协会联合主办的2023深圳中国1024程序员节开幕式暨主论坛活动在深圳龙华区启幕。以“领航鹏城发展,码动程序世界”为主题&a…...

打包和部署Java应用程序:Maven和Shell脚本的实用方法

在软件开发领域,高效打包和分发Java应用程序是至关重要的。本博客将探讨一种使用Maven插件和Shell脚本的简化方法,以创建一个分发包,其中包含了您项目的可执行JAR文件、配置文件和一个方便的启动脚本。 步骤1:Maven插件配置 旅程…...

Windows Python3安装salt模块失败处理

复现CVE-2020-11651时候运行CVE-2020-11651的poc时候需要salt模块 在下载时出现了错误 尝试在网上寻找解决方法: 1.更新 setuptools 和 wheel pip install --upgrade setuptools wheel 2. 安装Microsoft Visual C 14.0 因为salt模块包包使用了 C/C 扩展&#x…...

RabbitMQ 消息队列编程

安装与配置 安装 RabbitMQ 读者可以在 RabbitMQ 官方文档中找到完整的安装教程:Downloading and Installing RabbitMQ — RabbitMQ 本文使用 Docker 的方式部署。 RabbitMQ 社区镜像列表:https://hub.docker.com/_/rabbitmq 创建目录用于映射存储卷…...

基于安卓android微信小程序的个人管理小程序

运行环境 开发语言:Java 框架:ssm JDK版本:JDK1.8 服务器:tomcat7 数据库:mysql 5.7(一定要5.7版本) 数据库工具:Navicat11 开发软件:eclipse/myeclipse/idea Maven包&a…...

浏览器访问 AWS ECS 上部署的 Docker 容器(监听 80 端口)

✅ 一、ECS 服务配置 Dockerfile 确保监听 80 端口 EXPOSE 80 CMD ["nginx", "-g", "daemon off;"]或 EXPOSE 80 CMD ["python3", "-m", "http.server", "80"]任务定义(Task Definition&…...

设计模式和设计原则回顾

设计模式和设计原则回顾 23种设计模式是设计原则的完美体现,设计原则设计原则是设计模式的理论基石, 设计模式 在经典的设计模式分类中(如《设计模式:可复用面向对象软件的基础》一书中),总共有23种设计模式,分为三大类: 一、创建型模式(5种) 1. 单例模式(Sing…...

【杂谈】-递归进化:人工智能的自我改进与监管挑战

递归进化:人工智能的自我改进与监管挑战 文章目录 递归进化:人工智能的自我改进与监管挑战1、自我改进型人工智能的崛起2、人工智能如何挑战人类监管?3、确保人工智能受控的策略4、人类在人工智能发展中的角色5、平衡自主性与控制力6、总结与…...

Qt/C++开发监控GB28181系统/取流协议/同时支持udp/tcp被动/tcp主动

一、前言说明 在2011版本的gb28181协议中,拉取视频流只要求udp方式,从2016开始要求新增支持tcp被动和tcp主动两种方式,udp理论上会丢包的,所以实际使用过程可能会出现画面花屏的情况,而tcp肯定不丢包,起码…...

Appium+python自动化(十六)- ADB命令

简介 Android 调试桥(adb)是多种用途的工具,该工具可以帮助你你管理设备或模拟器 的状态。 adb ( Android Debug Bridge)是一个通用命令行工具,其允许您与模拟器实例或连接的 Android 设备进行通信。它可为各种设备操作提供便利,如安装和调试…...

在 Nginx Stream 层“改写”MQTT ngx_stream_mqtt_filter_module

1、为什么要修改 CONNECT 报文? 多租户隔离:自动为接入设备追加租户前缀,后端按 ClientID 拆分队列。零代码鉴权:将入站用户名替换为 OAuth Access-Token,后端 Broker 统一校验。灰度发布:根据 IP/地理位写…...

【CSS position 属性】static、relative、fixed、absolute 、sticky详细介绍,多层嵌套定位示例

文章目录 ★ position 的五种类型及基本用法 ★ 一、position 属性概述 二、position 的五种类型详解(初学者版) 1. static(默认值) 2. relative(相对定位) 3. absolute(绝对定位) 4. fixed(固定定位) 5. sticky(粘性定位) 三、定位元素的层级关系(z-i…...

ETLCloud可能遇到的问题有哪些?常见坑位解析

数据集成平台ETLCloud,主要用于支持数据的抽取(Extract)、转换(Transform)和加载(Load)过程。提供了一个简洁直观的界面,以便用户可以在不同的数据源之间轻松地进行数据迁移和转换。…...

现代密码学 | 椭圆曲线密码学—附py代码

Elliptic Curve Cryptography 椭圆曲线密码学(ECC)是一种基于有限域上椭圆曲线数学特性的公钥加密技术。其核心原理涉及椭圆曲线的代数性质、离散对数问题以及有限域上的运算。 椭圆曲线密码学是多种数字签名算法的基础,例如椭圆曲线数字签…...

图表类系列各种样式PPT模版分享

图标图表系列PPT模版,柱状图PPT模版,线状图PPT模版,折线图PPT模版,饼状图PPT模版,雷达图PPT模版,树状图PPT模版 图表类系列各种样式PPT模版分享:图表系列PPT模板https://pan.quark.cn/s/20d40aa…...