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

👋hi,我不是一名外包公司的员工,也不会偷吃茶水间的零食,我的梦想是能写高端CRUD
🔥 2025本人正在沉淀中… 博客更新速度++
👍 欢迎点赞、收藏、关注,跟上我的更新节奏
🎵 当你的天空突然下了大雨,那是我在为你炸乌云

文章目录

  • 一、入门
    • 什么是命令模式?
    • 为什么需要命令模式?
    • 怎样实现命令模式?
  • 二、命令模式在源码中的运用
    • 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;
    }

    @Override
    public void execute() {
        light.on();
    }
}

// 具体命令:关灯
class LightOffCommand implements Command {
    private Light light;

    public LightOffCommand(Light light) {
        this.light = light;
    }

    @Override
    public 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 {

	// 在这里可以加接收这
	
    @Override
    public 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 {

    @Override
    public 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);

        // 调用 CommandLineRunner
        callRunners(context, args);
    }

    private void callRunners(ApplicationContext context, String[] args) {
        // 获取所有 CommandLineRunner 的 Bean
        Map<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

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【DDD系列-3】DDD战术设计实践分享

DDD 战术设计概念​ ​ ​​ TMF2 中的概念&#xff1a;​ 领域能力&#xff1a;​ 扩展点&#xff1a;​ DDD 战术设计使用场景​ 复杂业务场景​ 复杂来源面对的需求方更加多样化。​ 1 相同场景不同垂直业务方的需求&#xff08;1688&#xff0c;淘宝&#xff0c;天…

基于单片机的仓库安防系统(论文+源码)

2.1 需求分析 仓库由于存有大量物品&#xff0c;因此对仓库的监控非常重要&#xff0c;目前仓库已经普遍装有安防系统&#xff0c;以保证仓库的安全&#xff0c;本次基于单片机的仓库安防系统设计&#xff0c;在功能上设计如下&#xff1a; 用户可通过IC卡进入仓库&#xff1…

使用 AutoMQ 和 Tinybird 分析用户网购行为

前言 在当前竞争激烈的市场环境中&#xff0c;数据分析已成为企业实现差异化和精准营销的关键。通过分析用户行为数据&#xff0c;企业能够深入了解用户的习惯、偏好和行为模式&#xff0c;从而更精准地定位目标市场&#xff0c;制定个性化营销策略&#xff0c;并提供定制化推…

小米 R3G 路由器刷机教程(Pandavan)

小米 R3G 路由器刷机教程&#xff08;Pandavan&#xff09; 一、前言 小米 R3G 路由器以其高性价比和稳定的性能备受用户青睐。然而&#xff0c;原厂固件的功能相对有限&#xff0c;难以满足高级用户的个性化需求。刷机不仅可以解锁路由器的潜能&#xff0c;还能通过第三方固…

Python数据可视化 - Matplotlib教程

文章目录 前言一、Matplotlib简介及安装1. Matplotlib简介2. 安装Matplotlib 二、Matplotlib Pyplot1. Pyplot介绍2. Pyplot中方法介绍2.1 创建和管理图形2.2 绘制图形2.3 设置图形属性2.4 保存和展示 三、Matplotlib绘图标记1. 介绍2. 基本用法3. 标记大小与颜色4. 标记样式列…

DeepSeek 与网络安全:AI 驱动的智能防御

&#x1f4dd;个人主页&#x1f339;&#xff1a;一ge科研小菜鸡-CSDN博客 &#x1f339;&#x1f339;期待您的关注 &#x1f339;&#x1f339; 1. 引言 随着人工智能&#xff08;AI&#xff09;的快速发展&#xff0c;深度学习技术正渗透到多个领域&#xff0c;从医疗诊断到…

STM32——HAL库开发笔记19(串口中断接收实验)(参考来源:b站铁头山羊)

本实验&#xff0c;我们以中断的方式使得串口发送数据控制LED的闪烁速度&#xff0c;发送1&#xff0c;慢闪&#xff1b;发送2&#xff0c;速度正常&#xff1b;发送3&#xff0c;快闪。 一、电路连接图 二、实现思路&CubeMx配置 1、实现控制LED的闪烁速度 uint32_t bli…

开关电源实战(一)宽范围DC降压模块MP4560

系列文章目录 文章目录 系列文章目录MP4560MP4560 3.8V 至 55V 的宽输入范围可满足各种降压应用 MOSFET只有250mΩ 输出可调0.8V-52V SW:需要低VF肖特基二极管接地,而且要靠近引脚,高压侧开关的输出。 EN:输入使能,拉低到阈值以下关闭芯片,拉高或浮空启动 COMP:Compens…

网络IP地址冲突故障,快速解决方案!

由于网络被广泛运用&#xff0c;网络规模持续变大&#xff0c;对应的 IP 地址分配也越来越多&#xff0c;IP 地址冲突的情况日益严重&#xff0c;在一定程度上对网络的正常运行造成了影响。 要维护网络稳定、高效地运行&#xff0c;解决 IP 地址冲突的问题就成了网络管理里的一…

C++模拟实现二叉搜索树

目录 1.二叉搜索树概念 2.二叉搜索树的实现 2.1二叉搜索树的查找 2.2二叉树的插入 2.3二叉树的删除 3.所有代码 4.二叉搜索树的应用 5.二叉搜索树的性能分析 1.二叉搜索树概念 二叉搜索树又称二叉排序树&#xff0c;它或者是一棵空树&#xff0c;或者是具有以下性质的二…

3D渐变柱状图

代码说明 数据准备&#xff1a; 数据可以是任意形式的矩阵&#xff0c;例如 5x7 的矩阵。 行标签 (rowLabels) 和列标签 (colLabels) 是可选的&#xff0c;如果不需要可以删除相关部分。 颜色定义&#xff1a; 使用自定义的蓝黄渐变色 (map)。 如果需要其他颜色&#xff0c;…

完美解决 error:0308010C:digital envelope routines::unsupported

查看专栏目录 Network 灰鸽宝典专栏主要关注服务器的配置&#xff0c;前后端开发环境的配置&#xff0c;编辑器的配置&#xff0c;网络服务的配置&#xff0c;网络命令的应用与配置&#xff0c;windows常见问题的解决等。 文章目录 windows电脑完美解决办法&#xff1a;设置说明…

Xilinx kintex-7系列 FPGA支持PCIe 3.0 吗?

Xilinx kintex-7系列资源如下图 Xilinx各系列的GT资源类型和性能 PCIe Gen1/2/3的传输速率对比 K7上面使用的高速收发器GTX最高速率为12.5GT/s&#xff0c; PCIe Gen2 每个通道的传输速率为 5 GT/s。 PCIe Gen3 每个通道的传输速率为 8 GT/s。 所以理论上硬件支持PCIe3.0&#…

支持列表拖拽嵌套,AI流式输出的多模态文档编辑器flowmix/docx: 全面升级

hi, 大家好, 我是徐小夕. 马上又到周五了, 最近也收到很多用户对 flowmix/docx 多模态文档编辑器的反馈&#xff0c;我们也做了一波新功能的升级&#xff0c;今天就和大家分享一下 flowmix/docx 多模态文档编辑器的最新更新. 演示地址: https://flowmix.turntip.cn/docx 以下是…

Mysql中使用sql语句生成雪花算法Id

&#x1f353; 简介&#xff1a;java系列技术分享(&#x1f449;持续更新中…&#x1f525;) &#x1f353; 初衷:一起学习、一起进步、坚持不懈 &#x1f353; 如果文章内容有误与您的想法不一致,欢迎大家在评论区指正&#x1f64f; &#x1f353; 希望这篇文章对你有所帮助,欢…

聊聊 IP 地址和端口号的区别

在计算机网络中&#xff0c;两个基本概念对于理解设备如何通过网络进行通信至关重要。IP 地址和端口号是 TCP/IP 的典型特征&#xff0c;其定义如下&#xff1a;IP 地址是分配给连接到网络的每台机器的唯一地址&#xff0c;用于定位机器并与其通信。相反&#xff0c;端口号用于…

【线性代数】1行列式

1. 行列式的概念 行列式的符号表示: 行列式的计算结果:一个数 计算模型1:二阶行列式 二阶行列式: 三阶行列式: n阶行列式: 🍎计算行列式 计算模型2:上三角形行列式 上三角形行列式特征:主对角线下皆为0。 上三角形行列式: 化上三角形通用方法:主对角线下,…

问界M8细节曝光,L3自动驾驶有了!

文 | AUTO芯球 作者 | 雷慢 太惊喜了&#xff0c; 问界M8近距离实拍曝光了&#xff0c; 我看了一圈&#xff0c; 给大家扒出几个炸裂的信息&#xff0c; 注意看侧身这一堆传感器&#xff0c; 这可不是什么普通摄像头&#xff0c; 这一片传感器和和尊界S800那套一模一样&a…

支付宝 IoT 设备入门宝典(上)设备管理篇

相信不少朋友最近都被支付宝“碰一下”广告刷屏&#xff0c;“不用打开 APP 支付就碰一下”几个字一出简直自带BGM……其实“碰一下”就是支付宝 IoT 设备的一种&#xff0c;趁着热度还在&#xff0c;我会分为设备管理和设备经营上下两篇&#xff0c;简单介绍一下支付宝 IoT&am…

【Linux网络-网络基础】计算机网络背景+协议+OSI七层模型

一、计算机网络背景 网络相关概念 1.什么是网络&#xff1f; 网络是一种由多个节点&#xff08;如计算机、手机或其他电子设备&#xff09;通过通信线路或无线信号连接而成的系统。在网络中&#xff0c;信息可以通过这些节点进行传输和交换 2.独立模式 独立模式&#xff1…