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

【设计模式】【行为型模式】命令模式(Command)

👋hi,我不是一名外包公司的员工,也不会偷吃茶水间的零食,我的梦想是能写高端CRUD
🔥 2025本人正在沉淀中… 博客更新速度++
📫 欢迎+V: flzjcsg2,我们共同讨论Java深渊的奥秘
🎵 当你的天空突然下了大雨,那是我在为你炸乌云

文章目录

  • 一、入门
    • 什么是命令模式?
    • 为什么需要命令模式?
    • 怎样实现命令模式?
  • 二、命令模式在源码中的运用
    • 2.1、JDK的Runnable接口
    • 2.2、Spring 的 CommandLineRunner 接口
  • 三、总结
    • 命令模式的优点
    • 命令模式的缺点
    • 命令模式的使用场景
  • 参考

一、入门

什么是命令模式?

命令模式是一种行为设计模式,它将请求或操作封装为对象,从而使你可以用不同的请求对客户进行参数化,并支持请求的排队、记录、撤销等操作。
命令模式的核心是将“请求”封装为独立的对象,包含执行操作所需的所有信息。这样,你可以将请求与执行者解耦,并通过参数化、队列或日志等方式管理请求。

为什么需要命令模式?

在没有使用命令模式的情况下,代码可能会遇到以下问题:

  • 紧耦合
    • 调用者(Invoker)直接依赖接收者(Receiver)的具体实现。如果接收者的接口或行为发生变化,调用者也需要修改。
    • 例如,一个按钮直接调用某个对象的特定方法,导致按钮代码与具体逻辑紧密耦合。
  • 难以扩展
    • 如果需要添加新的操作,必须修改调用者的代码,违反了开闭原则(对扩展开放,对修改关闭)。
    • 例如,一个遥控器需要支持新的设备时,必须修改遥控器的代码。
  • 不支持撤销、重做或事务操作
    • 如果系统需要支持撤销、重做或事务操作,直接调用方法的方式难以实现这些功能。
    • 例如,一个文本编辑器需要支持撤销操作,直接调用方法的方式无法记录历史状态。
  • 难以实现请求的队列或日志
    • 如果需要对请求进行排队、延迟执行或记录日志,直接调用方法的方式无法实现这些功能。

怎样实现命令模式?

命令模式的组成:
命令(Command):定义执行操作的接口,通常包含一个 execute() 方法。
具体命令(Concrete Command):实现命令接口,负责调用接收者的操作。
接收者(Receiver):实际执行操作的对象。
调用者(Invoker):持有命令对象,并触发命令的执行。
客户端(Client):创建命令对象并设置其接收者。

【案例】 开关灯
在这里插入图片描述
Light(接收者):实际执行操作的对象。包含 on()off()方法。’

class Light {public void on() {System.out.println("Light is ON");}public void off() {System.out.println("Light is OFF");}
}

Command(命令接口):定义执行操作的接口,包含 execute()方法。

interface Command {void execute();
}

LightOnCommandLightOffCommand(具体命令):实现 Command 接口,封装了对 Light 的操作。持有 Light 对象的引用,并在 execute() 方法中调用 Light 的方法。

// 具体命令:开灯
class LightOnCommand implements Command {private Light light;public LightOnCommand(Light light) {this.light = light;}@Overridepublic void execute() {light.on();}
}// 具体命令:关灯
class LightOffCommand implements Command {private Light light;public LightOffCommand(Light light) {this.light = light;}@Overridepublic void execute() {light.off();}
}

RemoteControl(调用者):持有 Command 对象的引用。通过pressButton()方法触发命令的执行。

class RemoteControl {private Command command;public void setCommand(Command command) {this.command = command;}public void pressButton() {command.execute();}
}

CommandPatternDemo(客户端):创建接收者、命令对象和调用者,并将它们组装在一起。

// 客户端
public class CommandPatternDemo {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();// 执行关灯命令remote.setCommand(lightOff);remote.pressButton();}
}

二、命令模式在源码中的运用

2.1、JDK的Runnable接口

Java 中的 Runnable 接口是命令模式的一个典型例子。
命令接口(类似于 Command 接口):Runnable接口。

public interface Runnable {void run();
}

具体命令(类似于 ConcreteCommand):我们自己实现的task,MyTask

public class MyTask implements Runnable {// 在这里可以加接收这@Overridepublic void run() {System.out.println("Task is running");}
}

调用者(类似于 Invoker): Thead类,下面是简化版

public class Thread {private Runnable target;public Thread(Runnable target) {this.target = target;}public void start() {task.run();}
}

客户端

public class Main {public static void main(String[] args) {Runnable task = new MyTask(); // 创建具体命令Thread thread = new Thread(task); // 设置命令thread.start(); // 执行命令}
}

2.2、Spring 的 CommandLineRunner 接口

CommandLineRunner 是 Spring 框架中一个非常有用的接口,通常用于在 Spring Boot 应用启动后执行一些初始化任务或自定义逻辑。它本质上是命令模式的一个典型应用,将“启动时需要执行的任务”封装为一个命令对象,并由 Spring Boot 在合适的时机统一执行。
CommandLineRunner 的作用:CommandLineRunner 接口的主要作用是在 Spring Boot 应用启动完成后,执行一些额外的逻辑。例如:初始化数据、加载配置文件、启动后台任务、执行一些检查或测试逻辑。
命令接口(类似于 Command 接口)CommandLineRunner

@FunctionalInterface
public interface CommandLineRunner {void run(String... args) throws Exception;
}

具体命令(类似于 ConcreteCommand)MyStartupTask

@Component // 将类注册为 Spring Bean
public class MyStartupTask implements CommandLineRunner {@Overridepublic void run(String... args) throws Exception {System.out.println("Executing startup task...");// 打印命令行参数System.out.println("Command line arguments:");for (String arg : args) {System.out.println(arg);}// 执行自定义逻辑initializeDatabase();loadConfiguration();}private void initializeDatabase() {System.out.println("Initializing database...");// 初始化数据库的逻辑}private void loadConfiguration() {System.out.println("Loading configuration...");// 加载配置文件的逻辑}
}

调用者(类似于 Invoker): SpringApplication,下面的代码是简化版

public class SpringApplication {public void run(String... args) {// 初始化 Spring 上下文ConfigurableApplicationContext context = createApplicationContext();refreshContext(context);// 调用 CommandLineRunnercallRunners(context, args);}private void callRunners(ApplicationContext context, String[] args) {// 获取所有 CommandLineRunner 的 BeanMap<String, CommandLineRunner> runners = context.getBeansOfType(CommandLineRunner.class);// 按顺序执行List<CommandLineRunner> sortedRunners = new ArrayList<>(runners.values());AnnotationAwareOrderComparator.sort(sortedRunners);// 调用每个 CommandLineRunner 的 run() 方法for (CommandLineRunner runner : sortedRunners) {runner.run(args);}}
}

客户端

@SpringBootApplication
public class MyApplication {public static void main(String[] args) {SpringApplication.run(MyApplication.class, args);}
}

三、总结

命令模式的优点

  • 解耦调用者和接收者:
    • 调用者(Invoker)不需要知道具体的接收者(Receiver)是谁,只需要调用命令对象的 execute() 方法。
    • 降低了系统的耦合度,使得调用者和接收者可以独立变化。
  • 支持扩展:
    • 可以轻松添加新的命令类,而不需要修改调用者的代码。
    • 符合开闭原则(对扩展开放,对修改关闭)。
  • 支持撤销和重做:
    • 命令对象可以记录状态,从而支持撤销(undo)和重做(redo)操作。
    • 例如,文本编辑器可以通过命令对象记录每次操作的状态,从而实现撤销功能。
  • 支持请求的队列或日志:
    • 命令对象可以被排队、延迟执行或记录日志。
    • 例如,可以将命令对象放入队列中,按顺序执行,或者将命令对象记录到日志中以便后续重放。
  • 支持事务操作:
    • 可以将多个命令组合成一个复合命令,实现事务操作。
    • 例如,在数据库操作中,可以将多个更新操作封装为一个事务。

命令模式的缺点

  • 类的数量增加:
    • 每个命令都需要一个具体的类,可能导致类的数量增多。
    • 对于简单的操作,使用命令模式可能会显得过于繁琐。
  • 复杂性增加:
    • 对于简单的请求,直接调用方法可能更直观,使用命令模式会增加额外的复杂性。
    • 需要额外的代码来管理命令对象(如队列、日志等)。

命令模式的使用场景

  • 需要解耦调用者和接收者:
    • 当调用者不需要知道接收者的具体实现时,可以使用命令模式。
    • 例如,GUI 中的按钮点击事件、远程调用的请求处理等。
  • 需要支持撤销、重做或事务操作:
    • 当系统需要支持撤销、重做或事务操作时,命令模式是一个很好的选择。
    • 例如,文本编辑器、绘图软件、数据库事务等。
  • 需要将请求排队或记录日志:
    • 当需要对请求进行排队、延迟执行或记录日志时,可以使用命令模式。
    • 例如,任务调度系统、消息队列、操作日志等。
  • 需要支持扩展:
    • 当系统需要支持新的操作,而不希望修改现有代码时,可以使用命令模式。
    • 例如,遥控器支持新的设备、插件系统等。

参考

黑马程序员Java设计模式详解, 23种Java设计模式(图解+框架源码分析+实战)_哔哩哔哩_bilibili

相关文章:

【设计模式】【行为型模式】命令模式(Command)

&#x1f44b;hi&#xff0c;我不是一名外包公司的员工&#xff0c;也不会偷吃茶水间的零食&#xff0c;我的梦想是能写高端CRUD &#x1f525; 2025本人正在沉淀中… 博客更新速度 &#x1f4eb; 欢迎V&#xff1a; flzjcsg2&#xff0c;我们共同讨论Java深渊的奥秘 &#x1f…...

C++模拟实现AVL树

目录 1.文章概括 2.AVL树概念 3.AVL树的性质 4.AVL树的插入 5.旋转控制 1.左单旋 2. 右单旋 3.左右双旋 4.右左双旋 6.全部代码 1.文章概括 本文适合理解平衡二叉树的读者阅读&#xff0c;因为AVL树是平衡二叉树的一种优化&#xff0c;其大部分实现逻辑与平衡二叉树是…...

推荐算法实践:movielens数据集

MovieLens 数据集介绍 MovieLens 数据集是由明尼苏达大学的GroupLens研究小组维护的一个广泛使用的电影评分数据集&#xff0c;主要用于推荐系统的研究。该数据集包含用户对电影的评分、标签以及其他相关信息&#xff0c;是电影推荐系统开发与研究的常用数据源。 数据集版本 …...

dynamic_cast和static_cast和const_cast

dynamic_cast 在 C 中的作用 dynamic_cast 是 C 运行时类型转换&#xff08;RTTI, Run-Time Type Identification&#xff09;的一部分&#xff0c;主要用于&#xff1a; 安全的多态类型转换检查类型的有效性向下转换&#xff08;Downcasting&#xff09;跨类层次的指针或引用…...

React进行路由跳转的方法汇总

在 React 中进行路由跳转有多种方法&#xff0c;具体取决于你使用的路由库和版本。以下是常见的路由跳转方法汇总&#xff0c;主要基于 react-router-dom 库。 1. 使用 useNavigate 钩子&#xff08;适用于 react-router-dom v6&#xff09; useNavigate 是 react-router-dom…...

python卷积神经网络人脸识别示例实现详解

目录 一、准备 1&#xff09;使用pytorch 2&#xff09;安装pytorch 3&#xff09;准备训练和测试资源 二、卷积神经网络的基本结构 三、代码实现 1&#xff09;导入库 2&#xff09;数据预处理 3&#xff09;加载数据 4&#xff09;构建一个卷积神经网络 5&#xff0…...

以Unity6.0为例,如何在Unity中开启DLSS功能

DLSS DLSS&#xff08;NVIDIA 深度学习超级采样&#xff09;&#xff1a;NVIDIA DLSS 是一套由 GeForce RTX™ Tensor Core 提供支持的神经渲染技术&#xff0c;可提高帧率&#xff0c;同时提供可与原生分辨率相媲美的清晰、高质量图像。目前最新突破DLSS 4 带来了新的多帧…...

CSDN 大模型 笔记

AI 3大范式&#xff1a;计算 发发 交互 L1 生成代码 复制到IDEA &#xff08;22年12-23年6&#xff0c;7月份&#xff09; L2 部分自动编程 定义class 设计interface 让其填充实现 (23年7&#xff0c;8月份) L3 通用任务 CRUD (24年) L4 高度自动编程 通用领域专有任务&#xf…...

Flink怎么保证Exactly - Once 语义

Exactly - Once 语义是消息处理领域中的一种严格数据处理语义&#xff0c;指每条数据都只会被精确消费和处理一次&#xff0c;既不会丢失&#xff0c;也不会重复。 以下从消息传递语义对比、实现方式、应用场景等方面详细介绍&#xff1a; 与其他消息传递语义对比 在消息传递…...

AOS安装及操作演示

文章目录 一、安装node1.1 在 macOS 上管理 Node版本1.1.1 安装 nvm1.1.2 验证 nvm 是否安装成功1.1.3 使用 nvm 安装/切换 Node.js 版本1.1.4 卸载 Node.js 版本 1.2 在 windows 上管理 Node版本1.2.1 安装 nvm-windows1.2.2 安装 Node.js 版本1.2.3 切换 Node.js 版本1.2.4 卸…...

Python 操作 MongoDB 教程

一、引言 在当今数字化时代&#xff0c;数据的存储和管理至关重要。传统的关系型数据库在处理一些复杂场景时可能会显得力不从心&#xff0c;而 NoSQL 数据库应运而生。MongoDB 作为一款开源的、面向文档的 NoSQL 数据库&#xff0c;凭借其高性能、高可扩展性和灵活的数据模型…...

Stability AI 联合 UIUC 提出单视图 3D 重建方法SPAR3D,可0.7秒完成重建并支持交互式用户编辑。

Stability AI 联合 UIUC 提出一种简单而有效的单视图 3D 重建方法 SPAR3D&#xff0c;这是一款最先进的 3D 重建器&#xff0c;可以从单视图图像重建高质量的 3D 网格。SPAR3D 的重建速度很快&#xff0c;只需 0.7 秒&#xff0c;并支持交互式用户编辑。 相关链接 论文&#xf…...

网易易盾接入DeepSeek,数字内容安全“智”理能力全面升级

今年农历新年期间&#xff0c;全球AI领域再度掀起了一波革命性浪潮&#xff0c;国产通用大模型DeepSeek凭借其强大的多场景理解与内容生成能力迅速“出圈”&#xff0c;彻底改写全球人工智能产业的格局。 作为国内领先的数字内容风控服务商&#xff0c;网易易盾一直致力于探索…...

自动驾驶---如何打造一款属于自己的自动驾驶系统

在笔者的专栏《自动驾驶Planning决策规划》中&#xff0c;主要讲解了行车的相关知识&#xff0c;从Routing&#xff0c;到Behavior Planning&#xff0c;再到Motion Planning&#xff0c;以及最后的Control&#xff0c;笔者都做了相关介绍&#xff0c;其中主要包括算法在量产上…...

局域网使用Ollama(Linux)

解决局域网无法连接Ollama服务的问题 在搭建和使用Ollama服务的过程中&#xff0c;可能会遇到局域网内无法连接的情况。经过排查发现&#xff0c;若开启了代理软件&#xff0c;尤其是Hiddify&#xff0c;会导致此问题。这一发现耗费了我数小时的排查时间&#xff0c;希望能给大…...

聚焦 AUTO TECH China 2025,共探汽车内外饰新未来Automotive Interiors

全球汽车产业蓬勃发展的大背景下&#xff0c;汽车内外饰作为汽车重要组成部分&#xff0c;其市场需求与技术创新不断推动着行业变革。2025年11月20日至22日&#xff0c;一场备受瞩目的行业盛会 ——AUTO TECH China 2025 广州国际汽车内外饰技术展览会将在广州保利世贸博览馆盛…...

Moretl 增量文件采集工具

永久免费: <下载> <使用说明> 用途 定时全量或增量采集工控机,电脑文件或日志. 优势 开箱即用: 解压直接运行.不需额外下载.管理设备: 后台统一管理客户端.无人值守: 客户端自启动,自更新.稳定安全: 架构简单,兼容性好,通过授权控制访问. 架构 技术架构: Asp…...

支持多种网络数据库格式的自动化转换工具——VisualXML

一、VisualXML软件介绍 对于DBC、ARXML……文件的编辑、修改等繁琐操作&#xff0c;WINDHILL风丘科技开发的总线设计工具——VisualXML&#xff0c;可轻松解决这一问题&#xff0c;提升工作效率。 VisualXML是一个强大且基于Excel表格生成多种网络数据库文件的转换工具&#…...

mysql8 用C++源码角度看客户端发起sql网络请求,并处理sql命令

MySQL 8 的 C 源码中&#xff0c;处理网络请求和 SQL 命令的流程涉及多个函数和类。以下是关键的函数和类&#xff0c;以及它们的作用&#xff1a; 1. do_command 函数 do_command 函数是 MySQL 服务器中处理客户端命令的核心函数。它从客户端读取一个命令并执行。这个函数在…...

四、OSG学习笔记-基础图元

前一章节&#xff1a; 三、OSG学习笔记-应用基础-CSDN博客https://blog.csdn.net/weixin_36323170/article/details/145514021 代码&#xff1a;CuiQingCheng/OsgStudy - Gitee.com 一、绘制盒子模型 下面一个简单的 demo #include<windows.h> #include<osg/Node&…...

使用vllm docker容器部署大语言模型

说明 最近deepseek比较火&#xff0c;我在一台4卡4090的服务器上尝试部署了一下&#xff0c;记录下部署步骤。 安装过程 安卓docker和nvidia-container-toolkit 安装19.03版本以上的docker-ce即可。安装步骤参考清华docker源上的安装步骤&#xff1a;Docker CE 软件仓库 为…...

window 安装GitLab服务器笔记

目录 视频&#xff1a; 资源&#xff1a; Linux CeneOS7&#xff1a; VMware&#xff1a; Linux无法安装 yum install vim -y 1.手动创建目录 2.下载repo PS 补充视频不可复制的代码 安装GitLab *修改root用户密码相关&#xff08;我卡在第一步就直接放弃了这个操作&…...

MySQL数据库入门到大蛇尚硅谷宋红康老师笔记 基础篇 part 10

第10章_创建和管理表 DDL&#xff1a;数据定义语言。CREATE \ALTER\ DROP \RENAME TRUNCATE DML&#xff1a;数据操作语言。INSERT \DELETE \UPDATE \SELECT&#xff08;重中之重&#xff09; DCL&#xff1a;数据控制语言。COMMIT \…...

react项目引入tailwindcss不生效解决方案

根据tailwindcss官网的操作步骤下来&#xff0c;样式未生效&#xff0c;且未报错&#xff0c;看了挺多的资料&#xff0c;还是并未解决。 后面在另一个项目尝试时&#xff0c;报了下面的问题&#xff1a; Error: PostCSS plugin tailwindcss requires PostCSS 8 根据这个链接…...

Expo运行模拟器失败错误解决(xcrun simctl )

根据你的描述&#xff0c;问题主要涉及两个方面&#xff1a;xcrun simctl 错误和 Expo 依赖版本不兼容。以下是针对这两个问题的解决方案&#xff1a; 解决 xcrun simctl 错误 错误代码 72 通常表明 simctl 工具未正确配置或路径未正确设置。以下是解决步骤&#xff1a; 确保 …...

【系统架构设计师】体系结构文档化

目录 1. 说明2. 重要性3. 主要内容4. 编写原则5. 实践建议6. 例题6.1 例题1 1. 说明 1.绝大多数的体系结构都是抽象的&#xff0c;由一些概念上的构建组成。2.层的概念在任何程序设计语言中都不存在。3.要让系统分析员和程序员去实现体系结构&#xff0c;还必须将体系结构进行…...

【0403】Postgres内核 检查(procArray )给定 db 是否有其他 backend process 正在运行

文章目录 1. 给定 db 是否有其他 backend 正在运行1.1 获取 allPgXact[] 索引1.1.1 MyProc 中 databaseId 初始化实现1.2 allProcs[] 中各 databaseId 判断1. 给定 db 是否有其他 backend 正在运行 CREATE DATABASE 语句创建用户指定 数据库名(database-name)时候, 会通过 …...

前端如何判断浏览器 AdBlock/AdBlock Plus(最新版)广告屏蔽插件已开启拦截

2个月前AdBlock/AdBlock Plus疑似升级了一次 因为自己主要负责面对海外的用户项目&#xff0c;发现以前的检测AdBlock/AdBlock Plus开启状态方法已失效了&#xff0c;于是专门研究了一下。并尝试了很多方法。 已失效的老方法 // 定义一个检测 AdBlock 的函数 function chec…...

微信小程序(第一集)

app.json {// 定义小程序的所有页面路径&#xff0c;数组中的第一个页面是首页"pages": ["pages/index/index", // 首页"pages/logs/logs" // 日志页面],// 设置小程序的全局窗口外观&#xff08;比如导航栏和背景颜色&#xff09;"wind…...

flutter ListView Item复用源码解析

Flutter 的 ListView 的 Item 复用机制是其高性能列表渲染的核心&#xff0c;底层实现依赖于 Flutter 的渲染管线、Element 树和 Widget 树的协调机制。以下是 ListView 复用机制的源码级解析&#xff0c;结合关键类和核心逻辑进行分析。 1. ListView 的底层结构 ListView 的复…...