设计模式-命令模式

一、定义

命令模式就是将一些请求封装为对象,以便使用不同的请求、队列、或者日志来参数化其他对象。命令模式也可以支持撤销的操作。

也就是说可以把一些动作封装为对象,以便于我们随心所欲地存储、传递和调用它们。

这种正式的定义一般都比较抽象的,我们下面通过设计一个遥控器的例子来理解。

二、实现

有这样一个需求,设计一个遥控器,遥控器上有很多插槽,这些插槽可以控制一些电器的开关,并且支持插槽的更换,比如插槽1原本控制灯,需要支持改成控制电视。而且这些电器对象的开和关的方法并不一致,比如灯的开关方法为on和off,电视的开关方法为open和close。

如何实现呢?

  1. 首先,肯定要解耦,支撑更换插槽这个操作,肯定不能把代码写死,就是我们不能在遥控器中直接调用电器的具体方法。
  2. 然后,每个电器的开关方法都不一样,而我们遥控器中使用的插槽对象肯定是一致的,所以说需要将这些电器的方法都包装到统一的命令类中。

其实也可以想象一下在饭店中点菜,我们都是在菜单上选中菜品之后,服务员把这个菜单交给厨师去做,我们不会直接和厨师交流,通过菜单实现了解耦。

定义命令接口和默认的无命令类

public interface Command {//执行方法void execute();
}public class NoCommand implements Command {@Overridepublic void execute() {System.out.println("没有命令");}
}

然后,定义具体的命令,每个电器的开和关的命令

//两个电器类
//电灯
public class Light {public void on(){System.out.println("打开了灯");}public void off(){System.out.println("关闭了灯");}
}
//电视
public class Television {public void open(){System.out.println("打开了电视");}public void close(){System.out.println("关闭了电视");}
}//电灯的命令类
public class LightOnCommand implements Command{private Light light;public LightOnCommand(Light light) {this.light = light;}@Overridepublic void execute() {light.on();}
}public class LightOffCommand implements Command{private Light light;public LightOffCommand(Light light) {this.light = light;}@Overridepublic void execute() {light.off();}
}//电视的命令类
public class TelevisionOnCommand implements Command{private Television television;public TelevisionOnCommand(Television television) {this.television = television;}@Overridepublic void execute() {television.open();}
}public class TelevisionOffCommand implements Command{private Television television;public TelevisionOffCommand(Television television) {this.television = television;}@Overridepublic void execute() {television.close();}
}

最后,来定义遥控器,这里就以只控制两个电器为例

public class RemoteControl {//默认控制两个电器private final int size = 2;private Command[] onCommands;private Command[] offCommands;public RemoteControl(){//创建遥控器时,插槽都是用默认的无命令对象Command noCommand = new NoCommand();onCommands = new Command[size];offCommands = new Command[size];for (int i = 0; i < size; i++) {onCommands[i] = noCommand;offCommands[i] = noCommand;}}public void setCommand(int index, Command onCommand, Command offCommand){if(index < 0 || index >= size){throw new RuntimeException("位置错误");}onCommands[index] = onCommand;offCommands[index] = offCommand;}public void pressOnButton(int index){if(index < 0 || index >= size){throw new RuntimeException("位置错误");}onCommands[index].execute();}public void pressOffButton(int index){if(index < 0 || index >= size){throw new RuntimeException("位置错误");}offCommands[index].execute();}
}

进行测试:

public class Test {public static void main(String[] args) {test();}static void test(){RemoteControl remoteControl = new RemoteControl();Light light = new Light();Television television = new Television();remoteControl.setCommand(0, new LightOnCommand(light), new LightOffCommand(light));remoteControl.setCommand(1, new TelevisionOnCommand(television), new TelevisionOffCommand(television));remoteControl.pressOnButton(0);remoteControl.pressOnButton(1);remoteControl.pressOffButton(0);remoteControl.pressOffButton(1);}
}
//输出
打开了灯
打开了电视
关闭了灯
关闭了电视

至此,我们通过设计统一的命令接口,将电器和遥控器进行解耦,遥控器不需要管怎么调用电器的开关方法,只需要调用插槽内的命令对象即可,这就是命令模式的简单实现。

三、增加撤销功能

我们想在遥控器上增加一个撤销按钮,按下去之后,就会撤销上一步操作。

其实对于我们现有的命令来说,所谓的撤销就是执行一次相反的操作嘛,比如撤销开灯的操作就需要执行一次关灯,那就可以修改一下Command接口类,新增一个撤销的方法,让各个命令类去实现具体的撤销操作,同时,遥控器需要保存上一步的操作是什么。

修改Command接口和各个实现类

//命令接口
public interface Command {//执行方法void execute();//撤销void undo();
}//电灯
public class LightOffCommand implements Command{private Light light;public LightOffCommand(Light light) {this.light = light;}@Overridepublic void execute() {light.off();}@Overridepublic void undo() {light.on();}
}
public class LightOnCommand implements Command{private Light light;public LightOnCommand(Light light) {this.light = light;}@Overridepublic void execute() {light.on();}@Overridepublic void undo() {light.off();}
}//电视
public class TelevisionOffCommand implements Command{private Television television;public TelevisionOffCommand(Television television) {this.television = television;}@Overridepublic void execute() {television.close();}@Overridepublic void undo() {television.open();}
}
public class TelevisionOnCommand implements Command{private Television television;public TelevisionOnCommand(Television television) {this.television = television;}@Overridepublic void execute() {television.open();}@Overridepublic void undo() {television.close();}
}

遥控器

public class RemoteControl {//默认控制两个电器private final int size = 2;private Command[] onCommands;private Command[] offCommands;private Command undoCommand;public RemoteControl(){Command noCommand = new NoCommand();onCommands = new Command[size];offCommands = new Command[size];for (int i = 0; i < size; i++) {onCommands[i] = noCommand;offCommands[i] = noCommand;}undoCommand = noCommand;}public void setCommand(int index, Command onCommand, Command offCommand){if(index < 0 || index >= size){throw new RuntimeException("位置错误");}onCommands[index] = onCommand;offCommands[index] = offCommand;}public void pressOnButton(int index){if(index < 0 || index >= size){throw new RuntimeException("位置错误");}onCommands[index].execute();undoCommand = onCommands[index];}public void pressOffButton(int index){if(index < 0 || index >= size){throw new RuntimeException("位置错误");}offCommands[index].execute();undoCommand = offCommands[index];}//撤销操作public void undo(){undoCommand.undo();}
}

测试:

public class Test {public static void main(String[] args) {test();}static void test(){RemoteControl remoteControl = new RemoteControl();Light light = new Light();Television television = new Television();remoteControl.setCommand(0, new LightOnCommand(light), new LightOffCommand(light));remoteControl.setCommand(1, new TelevisionOnCommand(television), new TelevisionOffCommand(television));remoteControl.pressOnButton(0);System.out.println("撤销上一步");remoteControl.undo();remoteControl.pressOnButton(0);remoteControl.pressOnButton(1);remoteControl.pressOffButton(0);System.out.println("撤销上一步");remoteControl.undo();remoteControl.pressOffButton(1);}
}
//输出
打开了灯
撤销上一步
关闭了灯
打开了灯
打开了电视
关闭了灯
撤销上一步
打开了灯
关闭了电视

通过测试我们看到我们成功实现了撤销功能。

四、复杂一点的撤销

上面的电器只是开关这两种完全相反的操作,撤销很容易实现,那如果是电风扇有不同的档位呢?如何实现?其实也很简单,只需要在设置档位的时候,记录下设置之前的档位是什么,撤销操作就是恢复到之前的档位。

并且一定要注意,这个记录操作也是同样放到命令类中的。

//风扇类
public class Fan {public static final int OFF = 0;public static final int LOWER = 1;public static final int MID = 2;public static final int HIGH = 3;private Integer level;public Fan() {this.level = OFF;}public Integer getLevel() {return level;}public void turnOFF(){this.level = OFF;}public void turnLower(){this.level = LOWER;}public void turnMid(){this.level = MID;}public void turnHigh(){this.level = HIGH;}
}//风扇的公共抽象命令类
public abstract class FanAbstractCommand implements Command{private Fan fan;private Integer lastLevel;public FanAbstractCommand(Fan fan) {this.fan = fan;}//子类去实现,子类在执行命令之前会记录风扇的当前状态到lastLevel中public abstract void execute();//撤销操作,根据记录的lastLevel进行撤销@Overridepublic void undo() {if(lastLevel == null) return;if(Fan.OFF == lastLevel){fan.turnOFF();}else if(Fan.LOWER == lastLevel){fan.turnLower();}else if(Fan.MID == lastLevel){fan.turnMid();}else if(Fan.HIGH == lastLevel){fan.turnHigh();}}public Fan getFan() {return fan;}public void setLastLevel(Integer lastLevel){this.lastLevel = lastLevel;}
}
//子类实现
public class FanLowerCommand extends FanAbstractCommand{public FanLowerCommand(Fan fan) {super(fan);}@Overridepublic void execute() {Fan fan = getFan();setLastLevel(fan.getLevel());fan.turnLower();}
}
public class FanMidCommand extends FanAbstractCommand{public FanMidCommand(Fan fan) {super(fan);}@Overridepublic void execute() {Fan fan = getFan();setLastLevel(fan.getLevel());fan.turnMid();}
}
//省略其他两个状态的子类

直接测试

public class Test {public static void main(String[] args) {testFan();}static void testFan(){RemoteControl remoteControl = new RemoteControl();//把0号插槽的开启位设置为低速,关闭位设置为关闭//把1号插槽的开启位设置为中速,关闭位设置为高速Fan fan = new Fan();remoteControl.setCommand(0, new FanLowerCommand(fan), new FanOffCommand(fan));remoteControl.setCommand(1, new FanMidCommand(fan), new FanHighCommand(fan));//测试remoteControl.pressOnButton(1);remoteControl.pressOffButton(1);System.out.println("撤销上一步");remoteControl.undo();}
}
//输出结果
风扇中速
风扇高速
撤销上一步
风扇中速

五、宏命令

现在的智能家居很多支持自定义宏,比如一键开启观影模式,就会打开电视、打开印象、关闭窗帘等一系列操作,我们有了命令模式要实现这种宏命令就很简单了。

你可能会想,直接定义一个命令,在这个命令的execute方法中直接调用一系列电器,这种方法可以实现,但是过于死板,比如我想在更改这个宏命令就只能修改代码,所以我们需要定义一个通用的宏命令,这个宏命令中存储一系列命令就好啦。

public class MacroCommand implements Command{private final List<Command> commands;public MacroCommand(List<Command> commands) {this.commands = commands;}@Overridepublic void execute() {for (Command command : commands) {command.execute();}}@Overridepublic void undo() {for (Command command : commands) {command.undo();}}
}

测试:

public class Test {public static void main(String[] args) {testMacro();}static void testMacro(){RemoteControl remoteControl = new RemoteControl();Light light = new Light();Television television = new Television();Fan fan = new Fan();List<Command> onCommands = Arrays.asList(new LightOnCommand(light),new TelevisionOnCommand(television),new FanLowerCommand(fan));MacroCommand macroOnCommand = new MacroCommand(onCommands);List<Command> offCommands = Arrays.asList(new LightOffCommand(light),new TelevisionOffCommand(television),new FanOffCommand(fan));MacroCommand macroOffCommand = new MacroCommand(offCommands);remoteControl.setCommand(0,macroOnCommand,macroOffCommand);System.out.println("===一键开启宏命令===");remoteControl.pressOnButton(0);System.out.println("===一键关闭宏命令===");remoteControl.pressOffButton(0);System.out.println("===撤销上一步操作===");remoteControl.undo();}
}//输出结果
===一键开启宏命令===
打开了灯
打开了电视
风扇低速
===一键关闭宏命令===
关闭了灯
关闭了电视
关闭风扇
===撤销上一步操作===
打开了灯
打开了电视
风扇低速

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

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

相关文章

第40天:Web开发-JS应用VueJS框架Vite构建启动打包渲染XSS源码泄露代码审计

#知识点 1、安全开发-VueJS-搭建启动&打包安全 2、安全开发-VueJS-源码泄漏&代码审计 一、Vue搭建创建项目启动项目 1、Vue 框架搭建->基于nodejs搭建&#xff0c;安装nodejs即可 参考&#xff1a;https://cn.vuejs.org/ 已安装18.3或更高版本的Node.js 2、Vue 创建…

DeepSeek做赛车游戏

赛车模型 2D生成图片 任意AI图片软件SD&#xff0c;MJ 图片生成3D模型 车身 车轮 场景 Rodin,Tripo和Meshy 询问deepSeek如何开发 拷贝代码 将汽车运行代码拖到汽车上 再让AI写个摄像头跟随代码 再去提问deepseek控制轮胎和一些处理细节

软考高级《系统架构设计师》知识点(一)

计算机硬件 校验码 码距&#xff1a;就单个编码A:00而言&#xff0c;其码距为1&#xff0c;因为其只需要改变一位就变成另一个编码。在两个编码中&#xff0c;从A码到B码转换所需要改变的位数称为码距&#xff0c;如A:00要转换为B:11&#xff0c;码距为2。一般来说&#xff0c;…

亚博microros小车-原生ubuntu支持系列:26手势控制小车基础运动

背景知识 手指检测&#xff1a;亚博microros小车-原生ubuntu支持系列&#xff1a;4-手部检测-CSDN博客 程序功能说明 功能开启后&#xff0c;摄像头捕获图像&#xff0c;识别手势来控制小车移动。 手势 “5”小车前进拳头小车后退手势 “1”小车向左手势 “2”小车向右 运…

OpenFeign远程调用返回的是List<T>类型的数据

在使用 OpenFeign 进行远程调用时&#xff0c;如果接口返回的是 List 类型的数据&#xff0c;可以通过以下方式处理&#xff1a; 直接定义返回类型为List Feign 默认支持 JSON 序列化/反序列化&#xff0c;如果服务端返回的是 List的JSON格式数据&#xff0c;可以直接在 Feig…

【hive】记一次hiveserver内存溢出排查,线程池未正确关闭导致

一、使用 MemoryAnalyzer软件打开hprof文件 很大有30G&#xff0c;win内存24GB&#xff0c;不用担心可以打开&#xff0c;ma软件能够生成索引文件&#xff0c;逐块分析内存&#xff0c;如下图。 大约需要4小时。 overview中开不到具体信息。 二、使用Leak Suspects功能继续…

【Docker】

一、概述 1、Docker为什么出现&#xff1f; 开发和运维两套环境&#xff0c;而环境配置十分麻烦。如在Windows上开发&#xff0c;要发布到Linux上运行。 Docker给以上问题提出解决方案&#xff1a;Java --- Jar(环境&#xff09;---打包项目带上环境&#xff08;镜像&#x…

游戏手柄Type-c方案,支持一边充电一边传输数据

乐得瑞推出LDR6023SS&#xff0c;专门针对USB-C接口手机手柄方案&#xff0c;支持手机快充&#xff0c;支持任天堂游戏机&#xff0c;PS4等设备~同时支持手机充电跟数据传输 1、概述 LDR6023SS SSOP16 是乐得瑞科技针对 USB Type-C 标准中的 Bridge 设备而开发的双 USB-C DRP …

【报错解决】Sql server 2022连接数据库时显示证书链是由不受信任的颁发机构颁发的

SSMS 20在连接Sql server 2022数据库时有如下报错&#xff1a; A connection was successfully established with the server, but then an error occurred during the login process. (provider: SSL Provider, error: 0 - 证书链是由不受信任的颁发机构颁发的。 原因是尝试使…

「vue3-element-admin」告别 vite-plugin-svg-icons!用 @unocss/preset-icons 加载本地 SVG 图标

&#x1f680; 作者主页&#xff1a; 有来技术 &#x1f525; 开源项目&#xff1a; youlai-mall ︱vue3-element-admin︱youlai-boot︱vue-uniapp-template &#x1f33a; 仓库主页&#xff1a; GitCode︱ Gitee ︱ Github &#x1f496; 欢迎点赞 &#x1f44d; 收藏 ⭐评论 …

NineData云原生智能数据管理平台新功能发布|2025年1月版

本月发布 14 项更新&#xff0c;其中重点发布 6 项、功能优化 7 项、安全性更新 1 项。 重点发布 数据库 Devops - 数据导出功能增强 支持 AWS ElastiCache 数据源&#xff1a;现已支持通过 SQL 查询语句或直接通过库表导出 AWS ElastiCache 数据&#xff0c;方便用户快速提取…

游戏引擎学习第96天

讨论了优化和速度问题&#xff0c;以便简化调试过程 节目以一个有趣的类比开始&#xff0c;提到就像某些高端餐厅那样&#xff0c;菜单上充满了听起来陌生或不太清楚的描述&#xff0c;需要依靠服务员进一步解释。虽然这听起来有些奇怪&#xff0c;但实际上&#xff0c;它反映…

Docker 1. 基础使用

1. Docker Docker 是一个 基于容器的虚拟化技术&#xff0c;它能够将应用及其依赖打包成 轻量级、可移植 的容器&#xff0c;并在不同的环境中运行。 2. Docker指令 &#xff08;1&#xff09;查看已有镜像 docker images &#xff08;2&#xff09;删除镜像 docker rmi …

基于机器学习时序库pmdarima实现时序预测

目录 一、Pmdarima实现单变量序列预测1.1 核心功能与特性1.2 技术优势对比1.3 python案例1.3.1 时间序列交叉验证1.3.1.1 滚动交叉验证1.3.1.2 滑窗交叉验证 时间序列相关参考文章&#xff1a; 时间序列预测算法—ARIMA 基于VARMAX模型的多变量时序数据预测 基于机器学习时序库…

【论文笔记】Are Self-Attentions Effective for Time Series Forecasting? (NeurIPS 2024)

官方代码https://github.com/dongbeank/CATS Abstract 时间序列预测在多领域极为关键&#xff0c;Transformer 虽推进了该领域发展&#xff0c;但有效性尚存争议&#xff0c;有研究表明简单线性模型有时表现更优。本文聚焦于自注意力机制在时间序列预测中的作用&#xff0c;提…

Matlab机械手碰撞检测应用

本文包含三个部分&#xff1a; Matlab碰撞检测的实现URDF文件的制作机械手STL文件添加夹爪 一.Matlab碰撞检测的实现 首先上代码 %% 检测在结构环境中机器人是否与物体之间发生碰撞情况&#xff0c;如何避免&#xff1f; % https://www.mathworks.com/help/robotics/ug/che…

从零开始:使用Jenkins实现高效自动化部署

在这篇文章中我们将深入探讨如何通过Jenkins构建高效的自动化部署流水线&#xff0c;帮助团队实现从代码提交到生产环境部署的全流程自动化。无论你是Jenkins新手还是有一定经验的开发者&#xff0c;这篇文章都会为你提供实用的技巧和最佳实践&#xff0c;助你在项目部署中走得…

鸿蒙harmony 手势密码

1.效果图 2.设置手势页面代码 /*** 手势密码设置页面*/ Entry Component struct SettingGesturePage {/*** PatternLock组件控制器*/private patternLockController: PatternLockController new PatternLockController()/*** 用来保存提示文本信息*/State message: string …

【Unity3D】UGUI的anchoredPosition锚点坐标

本文直接以实战去理解锚点坐标&#xff0c;围绕着将一个UI移动到另一个UI位置的需求进行说明。 &#xff08;anchoredPosition&#xff09;UI锚点坐标&#xff0c;它是UI物体的中心点坐标&#xff0c;以UI物体锚点为中心的坐标系得来&#xff0c;UI锚点坐标受锚点(Anchors Min…

Mp4视频播放机无法播放视频-批量修改视频分辨率(帧宽、帧高)

背景 家人有一台夏新多功能 视频播放器(夏新多功能 视频播放器),用来播放广场舞。下载了一些广场舞视频, 只有部分视频可以播放,其他视频均无法播放,判断应该不是帧速率和数据速率的限制, 分析可能是播放器不支持帧高度大于720的视频。由于视频文件较多,需要借助视频编…