设计模式—命令模式:探索【命令模式】的奥秘与应用实践!

命令模式

命令模式是一种行为设计模式,它的主要目的是将请求封装成一个对象,从而使得请求的发送者接收者之间进行解耦

在命令模式中,命令被封装为一个对象包含了需要执行的操作以及执行这些操作所需的所有参数

命令的发送者不需要知道接收者的具体情况,也不需要知道命令如何被执行,只需要将命令对象发送给接收者,由接收者来解析并执行命令。

应用场景

  • 请求者创建命令,消费者根据命令动态做出响应的场景
  • 撤销和恢复操作:例如文本编辑器中的撤销和恢复功能。
  • 请求日志记录:可以记录用户操作的日志,以便后期查看。
  • 任务队列:例如线程池中的任务队列,可以将命令放入队列中异步执行。
  • 菜单项操作:例如图形界面中的菜单项单击、快捷键操作等。

命令模式中的角色

  • 命令(Command): 命令是一个抽象的对象,封装了执行特定操作的方法。它通常包含一个执行操作的接口,可能还包含撤销操作的接口。
  • 具体命令(Concrete Command): 具体命令是命令的具体实现,它实现了命令接口中定义的方法。 每一个具体命令都与一个特定的接收者相关联,负责调用接收者执行请求的操作。
  • 接收者(Receiver): 接收者是实际执行命令操作的对象。它包含了命令执行时所需的具体逻辑。
  • 调用者/发送者(Invoker/Sender): 调用者是命令的发送者,它负责创建命令对象并将其发送给接收者。
  • 客户端(Client): 客户端创建具体命令对象,并将其与相应的接收者关联起来,然后将命令对象传递给调用者。

优点

  • **解耦:**发送者和接收者之间的解耦,发送者不需要知道接收者的具体实现,只需要知道如何发送命令。
  • **扩展性:**易于添加新的命令和接收者,不会影响到已有的代码。
  • **支持撤销和重做操作:**可以通过保存命令历史来支持撤销和重做操作。

缺点

  1. 可能导致类爆炸:为每个操作创建一个具体的命令类可能会导致类爆炸,尤其是在具有大量命令的系统中。
  2. 命令的逻辑混乱:如果命令的逻辑比较复杂,命令模式可能会导致命令的逻辑混乱。

Java中的注意事项

  1. 合理的接口设计:命令接口应该设计得简洁清晰,易于使用。
  2. 抽象类的使用:可以使用抽象类来实现部分命令共同的行为,以避免重复代码。
  3. 命令的实现方式:根据业务需求选择合适的命令实现方式,如简单命令、宏命令等。
  4. 线程安全性:在多线程环境下使用命令模式时,需要考虑命令对象的线程安全性。

1.案例1-小厨房(JavaSE版本)image-20240204144057601

# 餐厅,服务员先根据客户创建订单,并将订单发送给厨师,厨师根据具体的订单生产

1.1.创建命令接口

/*** 抽象命令接口*/
public interface CookCommand {/*** 执行命令*/void execute();
}

1.2.创建传输的参数

import lombok.Getter;
import lombok.Setter;
import java.util.ArrayList;
import java.util.List;@Getter @Setter
public class Order {// 订单数量private final List<Menu> orderList;public Order() {this.orderList = new ArrayList<Menu>();}// 创建订单public void createOrder(Menu menu){orderList.add(menu);}// 获取订单public List<Menu> getOrderList(){return orderList;}@Getter @Setterstatic class Menu{private String menuName; // 菜品名称private Integer num; // 菜品数量}}

1.3.创建接收者

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.lang.invoke.MethodHandles;
import java.util.List;/*** 命令接收人* @author 13723* @version 1.0* 2024/2/4 11:08*/
public class Cook {private static final Logger logger = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass());public void makeMenu(List<Order.Menu> menuList){logger.info("-------------------------> 厨师准备开始做菜 <-------------------------");for (Order.Menu menu : menuList) {logger.info("厨师正在做菜:{},数量:{}",menu.getMenuName(),menu.getNum());}logger.info("-------------------------> 厨师准备结束做菜 <-------------------------");}
}

1.4.创建具体命令

/*** 具体执行命令* @author 13723* @version 1.0* 2024/2/4 11:15*/
public class CookCommandImpl implements CookCommand{private static final Logger logger = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass());	// 厨师private Cook cook;// 订单private List<Order.Menu> orderList;CookCommandImpl(){}public CookCommandImpl(Cook cook,List<Order.Menu> orderList){this.cook = cook;this.orderList = orderList;}/*** 执行任务*/@Overridepublic void execute() {cook.makeMenu(orderList);}
}

1.5.创建调用者或者发送者

此处写在测试方法里了

/*** 命令模式 测试* @author 13723* @version 1.0* 2024/2/4 14:09*/
public class CommandTest {private static final Logger logger = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass());@Test@DisplayName("测试命令模式-点餐,上菜")public void  test1(){// 创建订单Order order = new Order();order.createOrder(new Order.Menu(){{setMenuName("土豆丝");setNum(1000);}});order.createOrder(new Order.Menu(){{setMenuName("拍黄瓜");setNum(5);}});order.createOrder(new Order.Menu(){{setMenuName("红烧肉");setNum(10);}});order.createOrder(new Order.Menu(){{setMenuName("黄焖鸡");setNum(30);}});// 发送命令到执行者CookCommandImpl cookCommand = new CookCommandImpl(new Cook(), order.getOrderList());// 执行命令cookCommand.execute();}
}

image-20240204144811442

2.案例2-撤销订单(SpringBoot版本)

# 我们假设我们有一个简单的订单管理系统,用户可以执行添加订单、删除订单等操作,并且希望能够撤销(bug很多没考虑)之前的操作。

2.1.定义注解

import org.springframework.stereotype.Service;
import java.lang.annotation.*;@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
@Service
public @interface Command {/*** 标记具体事务的业务类型*/String type()  default "";
}

2.2.定义订单命令接口

import com.hrfan.java_se_base.base_mvc.model.DecOrderHead;
/*** 定义订单命令接口*/
public interface OrderCommand {/*** 执行命令*/void execute(DecOrderHead order);/*** 撤销命令*/void undo(DecOrderHead order);
}

2.3.定义命令的具体实现类

import com.fasterxml.jackson.core.type.TypeReference;
import com.hrfan.java_se_base.base_mvc.mapper.DecOrderHeadMapper;
import com.hrfan.java_se_base.base_mvc.model.DecOrderHead;
import com.hrfan.java_se_base.base_mvc.service.DecOrderHeadService;
import com.hrfan.java_se_base.common.json.JsonObjectMapper;
import com.hrfan.java_se_base.pattern.commond_pattern.se.Order;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.stereotype.Service;
import org.springframework.util.Assert;import javax.annotation.Resource;
import java.lang.invoke.MethodHandles;
import java.util.Date;/*** 实现 添加订单命令 和 撤销订单命令 具体执行逻辑* @author 13723* @version 1.0* 2024/2/4 16:07*/
@Service
@Command(type = "insert")
public class InsertOrderCommand implements OrderCommand{private static final Logger logger = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass());@Resourceprivate DecOrderHeadMapper mapper;@Overridepublic void execute(DecOrderHead order) {logger.error("----------------- 添加操作开始 -----------------");logger.error("添加用户");// 留存备份(上一步操作前数据记录)String json = JsonObjectMapper.getInstance().toJson(order);order.setBeforeInfo(json);int insert = mapper.insert(order);logger.error(insert > 0 ? "添加成功!":"添加失败!");}/*** 撤销操作(此撤销 仅撤销一次,真正撤销需要考虑是否撤销到第一次,例如增加计数,改变+1 撤销-1 到为 1时,此时不能在撤销)* @param order 订单信息*/@Overridepublic void undo(DecOrderHead order) {logger.error("----------------- 撤销操作开始 -----------------");DecOrderHead decOrderHead = mapper.selectById(order.getSid());Assert.notNull(decOrderHead,"订单未找到,撤销失败!");logger.error("撤销用户!");String beforeInfo = order.getBeforeInfo();DecOrderHead tempOrderHead = JsonObjectMapper.getInstance().fromJson(beforeInfo, new TypeReference<DecOrderHead>() {});tempOrderHead.setBeforeTime(new Date());tempOrderHead.setBeforeUser("System");mapper.deleteById(order);int insert = mapper.insert(tempOrderHead);logger.error(insert > 0 ? "撤销成功!":"撤销失败!");}
}

2.4.定义命令管理类

import java.util.HashMap;
import java.util.List;
import java.util.Map;
import org.springframework.context.annotation.ClassPathScanningCandidateComponentProvider;
import org.springframework.core.annotation.AnnotationUtils;
import org.springframework.core.type.filter.AnnotationTypeFilter;/*** 定义命令管理器,用于将所有的使用注解的类注入到map中* 在使用的时候,直接从map中获取对应的命令,然后执行。*/
public class OrderCommandHistoryManager {/*** 存放 各种类型的具体命令* key :对应的业务乐星* value:对应的命令*/private Map<String,OrderCommand> handlerMap;public void setHandlerMap(List<OrderCommand> handlers) {handlerMap = scanHandlers(handlers);}/*** 扫描使用@Command注解的类 然后将其注入到map中* @param handlers 使用了@Command注解的类* @return 返回map集合*/private Map<String, OrderCommand> scanHandlers(List<OrderCommand> handlers) {ClassPathScanningCandidateComponentProvider scanner = new ClassPathScanningCandidateComponentProvider(false);scanner.addIncludeFilter(new AnnotationTypeFilter(Command.class));Map<String, OrderCommand> handlerMap = new HashMap<>();handlers.forEach((it) -> {Class<?> aClass = it.getClass();if (hasDutyAnnotation(aClass)) {// 判断注解上的类型是否填写String type = getOrderAnnotationValue(aClass);// 将类型放到map中handlerMap.put(type, it);}});return handlerMap;}/*** 判断该类是否是 使用使用了Command注解* @param clazz 类名称* @return true使用了*/private boolean hasDutyAnnotation(Class<?> clazz ) {return AnnotationUtils.findAnnotation(clazz, Command.class) != null;}/*** 判断使用了的注解@Command 是否填写了type 类型* @param clazz 类名称* @return 返回类型*/private String getOrderAnnotationValue(Class<?> clazz) {Command command = AnnotationUtils.findAnnotation(clazz, Command.class);if (command == null) {throw new IllegalStateException("Adapter annotation not found for class: " + clazz);}return command.type();}/*** 获取map集合* @return 返回map集合*/public Map<String,OrderCommand> getHandlerMap() {return handlerMap;}}

2.5.定义配置类

/*** 命令管理类的配置类,用于将OrderCommandHistoryManager 注入到Spring的Bean中*/
@Configuration
public class OrderCommandConfiguration {@Beanpublic OrderCommandHistoryManager handlerOrderCommandExecute(List<OrderCommand> handlers) {// 创建对象 会获取全部使用注解的类,将其注入到对应map集合中OrderCommandHistoryManager manager = new OrderCommandHistoryManager();manager.setHandlerMap(handlers);return manager;}
}

2.6.定义公共调用服务

import com.hrfan.java_se_base.base_mvc.model.DecOrderHead;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.util.Assert;
import java.lang.invoke.MethodHandles;/*** 定义命令模式的公共服务(将所有的命令在此处进行处理)* 将此方法对外暴露,供其他模块调用,隐藏具体的命令实现,只暴露接口* @author 13723* @version 1.0* 2024/2/4 17:45*/
@Service
public class CommandCommonExecuteService {private static final Logger logger = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass());@Autowiredprivate OrderCommandHistoryManager orderCommandHistoryManager;public void executeCommand(String type, DecOrderHead orderHead) {OrderCommand command = orderCommandHistoryManager.getHandlerMap().get(type);Assert.notNull(command,"no corresponding command found!");command.execute(orderHead);}public void undoLastCommand(String type,DecOrderHead orderHead) {OrderCommand command = orderCommandHistoryManager.getHandlerMap().get(type);Assert.notNull(command,"no corresponding command found!");command.undo(orderHead);}
}

2.7.测试订单管理

import com.hrfan.java_se_base.base_mvc.model.DecOrderHead;
import com.hrfan.java_se_base.common.json.JsonObjectMapper;
import com.hrfan.java_se_base.pattern.commond_pattern.boot.CommandCommonExecuteService;
import org.junit.jupiter.api.DisplayName;
import org.junit.jupiter.api.Test;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import java.lang.invoke.MethodHandles;
import java.util.Date;
import java.util.UUID;/*** @author 13723* @version 1.0* 2024/2/5 07:09*/
@SpringBootTest
public class OrderCommandTest {private static final Logger logger = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass());@Autowiredprivate CommandCommonExecuteService commandCommonService;@Test@DisplayName("测试命令模式")public void test(){logger.error("测试命令模式");DecOrderHead decOrderHead = new DecOrderHead() {{setSid(UUID.randomUUID().toString());setOrderNo("123456");setOrderPrice(100);setInsertTime(new Date());setInsertUser("Jack");}};// 添加订单前,将订单信息备份decOrderHead.setBeforeInfo(JsonObjectMapper.getInstance().toJson(decOrderHead));// 添加订单commandCommonService.executeCommand("insert",decOrderHead);// 撤销订单commandCommonService.undoLastCommand("insert",decOrderHead);}
}

image-20240205072515014

image-20240205074025556

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

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

相关文章

OpenGuass 之 where 1 = 0 处理流程代码走读

一. 前言 在OpenGuass中&#xff0c;如果where 条件中包含where 1 0 等固定为否条件的查询语句&#xff0c;在生成执行计划的时候&#xff0c;执行计划是BaseResult类型&#xff0c;此类型的执行计划不会进行物理数据扫描&#xff0c;如下所示&#xff1a; 对于非固定为否条件&…

【论文阅读】多传感器SLAM数据集

一、M2DGR 该数据集主要针对的是地面机器人&#xff0c;文章正文提到&#xff0c;现在许多机器人在进行定位时&#xff0c;其视角以及移动速度与车或者无人机有着较大的差异&#xff0c;这一差异导致在地面机器人完成SLAM任务时并不能直接套用类似的数据集。针对这一问题该团队…

latex中\documentclass[preprint,review,12pt]{elsarticle}的详细解释

在LaTeX中&#xff0c;\documentclass 是一个命令&#xff0c;用于指定文档所使用的文档类。文档类定义了文档的总体结构、格式和样式。elsarticle 是一个常用的文档类&#xff0c;它主要用于在Elsevier出版的期刊上提交论文。 详细解释 \documentclass[preprint,review,12pt…

Autosar Appl介绍

AUTOSAR架构中的应用层 AUTOSAR 应用层构成AUTOSAR 架构中的最顶层,被认为对所有车辆应用至关重要。AUTOSAR 标准使用“组件”概念指定应用层实现。 在谈论应用层实现时,应该考虑的三个最重要的部分是: AUTOSAR 应用软件组件这些组件的 AUTOSAR 端口AUTOSAR 端口接口 AUTOS…

浙江大学主办!2024年第7届信息通信与信号处理国际会议( ICICSP2024)征稿开启!

会议官网 IEEE | ICICSP 2024 学术会议查询-学术会议交流服务平台-爱科会易 (uconf.com)​www.uconf.com/

OpenChat:性能高达105.7%,第一个超越ChatGPT的开源模型?

OpenChat&#xff1a;性能高达105.7%&#xff0c;第一个超越ChatGPT的开源模型&#xff1f; 前几天开源模型第一还是是Vicuna-33B、WizardLM&#xff0c;这不又换人了。对于开源模型的风起云涌&#xff0c;大家见怪不怪&#xff0c;不断更新的LLM榜单似乎也没那么吸引人了。 …

在springboot项目中调用通义千问api多轮对话并实现流式输出

官网文档 阿里灵积提供了详细的官方文档 如何实现多轮对话 官方文档中提到只需要把每轮对话中返回结果添加到消息管理器中&#xff0c;就可以实现多轮对话。本质上就是将历史对话再次发送给接口。 如何实现流式输出 官方文档中提出使用streamCall()方法就可以实现流式输出&…

ViT的若干细节

之前只看了ViT的大概结构&#xff0c;具体的模型细节和代码实现知之甚少。随着ViT逐渐成为CV领域的backbone&#xff0c;有必要重新审视下。 patch -> token 为了将图片处理成序列格式&#xff0c;很自然地想到将图片分割成一个个patch&#xff0c;再把patch处理成token。 …

Linux学习:初始Linux

目录 1. 引子&#xff1a;1.1 简述&#xff1a;操作系统1.2 学习工具 2. Linux操作系统中的一些基础概念与指令2.1 简单指令2.2 ls指令与文件2.3 cd指令与目录2.4 文件目录的新建与删除指令2.5 补充指令1&#xff1a;2.6 文件编辑与拷贝剪切2.7 文件的查看2.8 时间相关指令2.9 …

【网站项目】202物流管理系统

&#x1f64a;作者简介&#xff1a;拥有多年开发工作经验&#xff0c;分享技术代码帮助学生学习&#xff0c;独立完成自己的项目或者毕业设计。 代码可以私聊博主获取。&#x1f339;赠送计算机毕业设计600个选题excel文件&#xff0c;帮助大学选题。赠送开题报告模板&#xff…

不会代码的时候,如何使用Jmeter完成接口测试

1.接口测试简介 接口测试是测试系统组件间接口的一种测试。接口测试主要用于检测外部系统与系统之间以及内部各个子系统之间的交互点。测试的重点是要检查数据的交换&#xff0c;传递和控制管理过程&#xff0c;以及系统间的相互逻辑依赖关系等。 2.接口测试流程 接口测试的…

2024.3.1 小项目

1、机械臂 #include <myhead.h> #define SER_IP "192.168.125.32" //服务器端IP #define SER_PORT 8888 //服务器端端口号#define CLI_IP "192.168.68.148" //客户端IP #define CLI_PORT 9999 /…

Python matplotlib

目录 1、安装 matplotlib 2、绘制折线图 修改标签文字和线条粗细 校正图形 3、绘制散点图 绘制单点 绘制一系列点 自动计算数据 删除数据点的轮廓 自定义颜色 使用颜色映射 自动保存图表 4、随机漫步 创建 RandomWalk() 类 选择方向 绘制随机漫步图 给点着色 …

最简单的ubuntu远程桌面方法

最简单的ubuntu远程桌面方法 部署环境&#xff1a;Ubuntu 20.04 LTS 现在最常用的远程控制Linux系统的方法是通过XRDP、VNC等&#xff0c;但是安装配置过程繁琐复杂&#xff0c;经常出现各种问题导致连接失败&#xff0c;另外一方面延迟较高&#xff0c;操作卡顿。 经过我坚…

【Java项目介绍和界面搭建】拼图小游戏——键盘、鼠标事件

&#x1f36c; 博主介绍&#x1f468;‍&#x1f393; 博主介绍&#xff1a;大家好&#xff0c;我是 hacker-routing &#xff0c;很高兴认识大家~ ✨主攻领域&#xff1a;【渗透领域】【应急响应】 【Java】 【VulnHub靶场复现】【面试分析】 &#x1f389;点赞➕评论➕收藏 …

DDS数据分发服务——提升汽车领域数据传输效率

1.引言 随着智能化技术的快速发展&#xff0c;汽车行业正经历着一场革命性的变革。如今的分布式系统变得越来越复杂且庞大&#xff0c;对网络通信基数要求在功能和性能层面越来越高。数据分发服务&#xff08;DDS&#xff09;作为一项先进的数据传输解决方案&#xff0c;在汽车…

2369. 检查数组是否存在有效划分(动态规划)

2024-3-1 文章目录 [2369. 检查数组是否存在有效划分](https://leetcode.cn/problems/check-if-there-is-a-valid-partition-for-the-array/)思路&#xff1a;代码&#xff1a; 2369. 检查数组是否存在有效划分 思路&#xff1a; 1.状态定义:f[i]代表考虑将[0,i]是否能被有效划…

电脑要用多少V的电源?电脑电源输入电压是市电

台式电源的输出电压是多少&#xff1f; 电脑电源输出一般有三种不同的电压&#xff0c;分别是&#xff1a; 12V、5V、3.3V。 电脑电源负责给电脑配件供电&#xff0c;如CPU、主板、内存条、硬盘、显卡等&#xff0c;是电脑的重要组成部分。 工作电流根据不同的硬件及其使用状…

Python算法100例-3.3 阿姆斯特朗数

完整源代码项目地址&#xff0c;关注博主私信源代码后可获取 1.问题描述2.问题分析3.算法设计4.确定程序框架5.完整的程序6.问题拓展 1&#xff0e;问题描述 如果一个整数等于其各个数字的立方和&#xff0c;则该数称为“阿姆斯特朗数”&#xff08;亦称为自恋性数&#xff…

nacos开启鉴权+springboot配置用户名密码

nacos默认没有开启鉴权&#xff0c;springboot无需用户名密码即可连接nacos。从2.2.2版本开始&#xff0c;默认控制台也无需登录直接可进行操作。 因此本文记录一下如何开启鉴权&#xff0c;基于nacos2.3.0版本。 编辑nacos服务端的application.properties&#xff1a; # 开…