设计模式之 命令模式

devtools/2024/11/25 7:43:36/

命令模式(Command Pattern)是行为型设计模式之一,它将请求(或命令)封装成一个对象,从而使用户能够将请求发送者与请求接收者解耦。通过命令模式,调用操作的对象与执行操作的对象不直接关联,命令对象作为一个中介,传递请求和执行的责任,使得用户可以通过简单的调用接口来执行复杂的行为。

一、命令模式的基本概念

命令模式的核心思想是将请求封装为一个对象,通常这个对象包含了执行某个特定操作的必要信息和执行的逻辑。当请求被调用时,该命令对象会通知相应的执行者对象执行特定的操作。通过这种方式,命令的调用者和执行者之间无需直接交互,符合松耦合的设计原则。

1. 主要组成部分

命令模式通常包括以下几个关键角色:

  • Command(命令接口):声明了执行命令的接口,通常包含一个execute()方法。

  • ConcreteCommand(具体命令类):实现了命令接口,封装了与特定请求相关的动作,并将请求转发给接收者对象。

  • Receiver(接收者类):执行与请求相关的具体操作。它知道如何执行与请求相关的操作。

  • Invoker(调用者类):请求命令的发送者,它持有一个命令对象,并在适当的时候调用该命令对象的execute()方法。

  • Client(客户端):客户端创建一个具体的命令对象,并设置其接收者。客户端通常还会将该命令对象传递给调用者。

2. 工作原理

命令模式的工作原理是,当客户端希望执行某个操作时,它创建一个命令对象(通常是ConcreteCommand的实例),并将该对象交给调用者。调用者在适当的时候调用execute()方法,命令对象将请求委托给接收者进行实际的操作。由于所有的请求都通过命令对象封装并传递,调用者与接收者完全解耦。

二、命令模式的代码示例

  • 命令接口
    public interface Command {void execute();
    }
  • 具体命令类
    public class OrderCommand implements Command{private Chef chef;private Order order;public OrderCommand(Chef chef, Order order) {this.chef = chef;this.order = order;}@Overridepublic void execute() {System.out.println(order.getDiningTable()+"桌的餐");Set<String> foodNames = order.getFoodDesc().keySet();for (String foodName : foodNames) {chef.makeFood(order.getFoodDesc().get(foodName), foodName);}}
    }
  • 厨师
    public class Chef {public void makeFood(int number, String foodName){System.out.println("正在制作"+number+"份"+foodName);}
    }
    
  • 调用者类
    public class Waitor {private List<Command> commands = new ArrayList<>();public void setCommand(Command command){commands.add(command);}public void OrderUp(){commands.forEach(command -> {command.execute();});}
    }
  • 订单实体类
    public class Order {private int diningTable;Map<String,Integer> foodDesc = new HashMap<>();public int getDiningTable() {return diningTable;}public void setDiningTable(int diningTable) {this.diningTable = diningTable;}public Map<String, Integer> getFoodDesc() {return foodDesc;}public void setFoodDesc(String name,Integer num) {foodDesc.put(name, num);}
    }
  • 客户端代码
    public class Client {public static void main(String[] args) {Order order = new Order();order.setDiningTable(1);order.setFoodDesc("西红柿炒蛋", 2);order.setFoodDesc("拉面", 2);Command command = new OrderCommand(new Chef(), order);command.execute();}
    }
  • 输出结果

三、命令模式的优缺点

优点:
  1. 解耦请求者与执行者:调用者(Invoker)不需要知道具体的执行操作,只需要通过命令接口进行调用,避免了直接依赖具体的执行类。请求者和执行者之间的耦合度降低,系统更灵活。
  2. 支持命令的撤销与恢复命令模式可以很方便地实现命令的撤销(Undo)和恢复(Redo)功能。通过保存历史命令,可以在需要时撤销之前的操作。
  3. 增加新的命令:添加新的命令非常简单,只需要定义一个新的命令类并实现Command接口,不需要修改现有的类或调用者,符合开闭原则。
  4. 组合命令:多个命令可以组合成一个“复合命令”,通过调用一个复合命令来依次执行多个操作,方便管理多个命令的执行。
缺点:
  1. 类的数量增加命令模式会导致系统中增加很多命令类,特别是在命令种类多的情况下,可能会使系统变得较为复杂。
  2. 可能不必要的复杂度:对于一些简单的应用,命令模式可能会引入不必要的复杂度。在这些情况下,直接在客户端调用相关方法可能更加简洁。
  3. 客户端需要了解命令对象:虽然命令模式解耦了请求者与执行者,但客户端需要了解如何配置和使用命令对象,这可能增加一些学习和使用的成本。

四、命令模式的应用场景

命令模式特别适用于以下几种场景:

  1. GUI(图形用户界面)按钮操作

    在GUI应用中,用户点击按钮通常会触发一些行为,命令模式非常适合将按钮的操作封装为命令对象,解耦按钮与具体操作的实现。通过命令对象,可以将按钮的功能独立出来,方便修改和扩展。
  2. 事务管理

    对于需要管理多个操作的系统,命令模式可以用来记录每个操作(命令),并且在需要时执行或撤销这些操作。比如在事务处理系统中,可以使用命令模式来实现事务的提交、回滚操作。
  3. 宏命令

    如果某个操作需要执行一系列操作,可以通过将多个命令对象组合成一个复合命令(Macro Command),从而一次性执行一组操作。
  4. 日志系统

    在一些系统中,操作可能需要被记录(例如,用户在系统中执行了某个动作)。可以将每个操作封装成命令对象,并在执行时记录这些操作,甚至可以在以后重新执行。
  5. 远程操作系统

    在需要远程控制设备的场景中,命令模式可以帮助将操作封装成命令对象,通过网络发送命令对象,从而远程执行设备操作。

http://www.ppmy.cn/devtools/136787.html

相关文章

解决 Gradle 报错:`Plugin with id ‘maven‘ not found` 在 SDK 开发中的问题

在 SDK 开发过程中&#xff0c;使用 Gradle 构建和发布 SDK 是常见的任务。在将 SDK 发布为 AAR 或 JAR 包时&#xff0c;你可能会使用 apply plugin: maven 来发布到本地或远程的 Maven 仓库。但是&#xff0c;随着 Gradle 版本的更新&#xff0c;特别是从 Gradle 7 版本开始&…

用源码编译虚幻引擎,并打包到安卓平台

用源码编译虚幻引擎&#xff0c;并打包到安卓平台 前往我的博客,获取更优的阅读体验 作业内容: 源码编译UE5.4构建C项目&#xff0c;简单设置打包到安卓平台 编译虚幻 5 前置内容 这里需要将 Epic 账号和 Github 账号绑定&#xff0c;然后加入 Epic 邀请的组织&#xff0c…

springboot 整合 rabbitMQ (延迟队列)

前言&#xff1a; 延迟队列是一个内部有序的数据结构&#xff0c;其主要功能体现在其延时特性上。这种队列存储的元素都设定了特定的处理时间&#xff0c;意味着它们需要在规定的时间点或者延迟之后才能被取出并进行相应的处理。简而言之&#xff0c;延时队列被设计用于存放那…

输入/输出管理 III(磁盘和固态硬盘)

一、磁盘 【总结】&#xff1a; 磁盘&#xff08;Disk&#xff09;是由表面涂有磁性物质的物理盘片&#xff0c;通过一个称为磁头的导体线圈从磁盘存取数据。在读&#xff0f;写操作期间&#xff0c;磁头固定&#xff0c;磁盘在下面高速旋转。如下图所示&#xff1a; 磁盘盘面…

WireShark

1. WireShark安装 去官网Wireshark Download下载安装包 之后按照安装向导进行安装。安装完成后&#xff0c;用户可以在桌面或应用程序中找到Wireshark的图标&#xff0c;双击打开软件 2. wireshark过滤语法 2.1过滤协议 http 2.2 过滤指定ip 匹配指定的IP地址数据包&…

i春秋-文件包含绕过(PHP伪协议的使用)

练习平台地址 竞赛中心 题目描述 题目内容 本地包含&#xff0c;进行BP爆破 成功获得了数据库敏感文件&#xff08;配置文件和/etc/passwd&#xff09; 但是发现/etc/shadow文件无法访问&#xff0c;那么无法登录数据库 尝试使用PHP伪协议读取文件 php://filter/resourceflag…

DTO和VO的区别及使用场景详解

随着互联网的发展&#xff0c;前后端分离的开发模式越来越流行。在前后端数据交互过程中&#xff0c;为了保证数据的安全性和效率&#xff0c;通常会采用 DTO 和 VO 来封装数据。本篇博客将详细介绍 DTO 和 VO 的区别以及使用场景。 大家可能会有个疑问&#xff0c;既然DTO是展…

微信小程序中的WXSS与CSS的关系及使用技巧

微信小程序中的WXSS与CSS的关系及使用技巧 引言 在微信小程序的开发中,样式的设计与实现是构建用户友好界面的关键。微信小程序使用WXSS(WeiXin Style Sheets)作为其样式表语言,WXSS在语法上与CSS非常相似,但也有一些独特的特性。本文将深入探讨WXSS与CSS的关系,介绍WX…