设计模式之命令模式

devtools/2025/1/31 4:30:40/

一、详细介绍

        命令模式是一种行为型设计模式它将“请求”封装为一个对象,使得使用请求、参数化请求、队列请求、撤销请求、日志请求等多种请求变得简单。命令模式通过将“行为请求者”与“行为实现者”解耦,使得请求的发送者和接收者完全分离,实现命令的发送者与执行者之间的解耦。

命令模式包含以下几个关键角色

  1. 命令接口(Command):定义命令的公共接口,声明执行命令的抽象方法。

  2. 具体命令(Concrete Command):实现命令接口,绑定一个接收者对象,并实现命令接口中声明的执行方法,负责调用接收者的相应操作方法。

  3. 接收者(Receiver):负责执行命令请求的具体操作,是命令的真正执行者。

  4. 调用者(Invoker):负责调用命令对象的执行方法,与命令对象之间通过命令接口交互,不关心命令的具体实现。

二、使用场景

  1. 需要将请求调用者与请求接收者解耦:通过命令模式,调用者无需了解接收者的具体实现,只需调用命令对象的执行方法。

  2. 需要支持命令的撤销/重做操作:命令对象可以记录其执行状态,方便实现撤销和重做功能。

  3. 需要支持命令队列:可以将命令对象放入队列中,按照一定顺序依次执行。

  4. 需要支持日志记录、事务操作等:通过命令对象记录操作历史,便于回溯和审计。

三、注意事项

  1. 命令接口的设计:命令接口应尽可能简洁,只包含必要的执行方法,避免引入过多的复杂性。

  2. 命令对象的生命周期管理:确保在适当的时候创建、执行、撤销命令对象,避免资源泄露。

  3. 命令对象的线程安全:如果命令对象在多线程环境中使用,需要考虑线程安全问题。

四、优缺点

优点:

  1. 降低对象之间的耦合度:调用者与接收者之间通过命令对象进行交互,二者之间无需直接依赖。

  2. 支持命令的撤销/重做:命令对象可以记录其执行状态,便于实现撤销和重做功能。

  3. 易于扩展新的命令:只需增加新的具体命令类,符合开闭原则。

缺点:

  1. 命令类数量可能会过多:如果系统中命令类型较多,可能会导致命令类数量增长较快。

  2. 命令模式可能会增加系统的复杂性:引入额外的命令、接收者、调用者等类,增加了系统的理解和维护难度。

五、Java代码示例

以下是一个简单的Java代码示例,展示了使用命令模式控制家电设备:

// 命令接口(Command)
interface Command {void execute();
}// 具体命令(Concrete Command)
class TurnOnCommand implements Command {private final ElectricDevice device;public TurnOnCommand(ElectricDevice device) {this.device = device;}@Overridepublic void execute() {device.turnOn();}
}class TurnOffCommand implements Command {private final ElectricDevice device;public TurnOffCommand(ElectricDevice device) {this.device = device;}@Overridepublic void execute() {device.turnOff();}
}// 接收者(Receiver)
class ElectricDevice {public void turnOn() {System.out.println("Electric device turned on.");}public void turnOff() {System.out.println("Electric device turned off.");}
}// 调用者(Invoker)
class RemoteControl {private Command command;public void setCommand(Command command) {this.command = command;}public void pressButton() {command.execute();}
}// 客户端代码
public class Client {public static void main(String[] args) {ElectricDevice device = new ElectricDevice();RemoteControl remote = new RemoteControl();Command turnOnCommand = new TurnOnCommand(device);Command turnOffCommand = new TurnOffCommand(device);remote.setCommand(turnOnCommand);remote.pressButton(); // Output: Electric device turned on.remote.setCommand(turnOffCommand);remote.pressButton(); // Output: Electric device turned off.}
}

六、问题与解决方案

  1. 命令对象过多导致管理困难:可以使用工厂模式、注册表模式等来集中管理和创建命令对象。

  2. 命令对象的撤销/重做功能实现复杂:可以为命令接口添加undo()redo()方法,并在具体命令中实现这些方法。如果撤销/重做操作较复杂,可以考虑使用备忘录模式辅助实现。

七、与其他模式的对比

  1. 与策略模式的对比:策略模式强调算法或行为的替换,命令模式强调请求的封装和解耦。策略模式中的上下文直接使用策略对象,而命令模式中的调用者通过命令对象间接调用接收者。

  2. 与观察者模式的对比:观察者模式关注对象状态变化的通知,命令模式关注请求的封装和解耦。观察者模式中观察者被动接收通知,命令模式中调用者主动调用命令。

  3. 与模板方法模式的对比:模板方法模式定义了算法骨架,子类填充具体步骤。命令模式中命令对象执行的是完整的行为请求,而非算法的一部分。


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

相关文章

23种设计模式

说明:今天学习了几种设计模式,浅浅的记录一下,方便以后查看复习的同时希望也能帮助到有同样需求的同学。 1、建造者模式 这个模式我之前实习的时候遇到过了,天天build build的当时也不知道啥意思,今天终于弄明白了。按…

Linux:使用ssl加密网站为https

Linux:使用ssl加密网站为https 生成对应的数字证书以及密钥 [rootserver100 ~]# openssl req -newkey rsa:2048 -nodes -sha256 -keyout /etc/nginx/certs/timinglee.org.key -x509 -days 365 -out /etc/nginx/certs/timinglee.org.crt......*...................…

nodejs 中间件

一、是什么 Node.js 中的中间件,特别是针对 Web 开发框架(如 Express、Koa、Hapi 等)的中间件,其核心功能是用来对 HTTP 请求生命周期进行拦截、处理和传递的。 中间件这一概念是 Web 开发框架为了实现请求处理流程的模块化、可…

Ansible一键部署zabbix+grafana+agent

目录 IP地址规划ansible安装分开部署安装zabbix-mysql安装zabbix-server安装zabbix-agent安装zabbix-grafana 一键部署自动发现 IP地址规划 名字地址主要安装软件ansible-server192.168.40.137zabbix-server、ansible、zabbix-mysqlzabbix-agent1192.168.40.138zabbix-agentza…

Django杂记之数据查询和文件处理

Hello , 我是小恒不会java。本文内容可能比较杂,主要是一些重点片段,不会涉及真实业务 学过django的读者可以放心阅读,本文约2万字,阅读时间不会很长 在线网址阅读http://django.yunduanjianzhan.cn/index. 数据查询 查询 想要…

算法训练营day24

参考链接代码随想录 (programmercarl.com) 一、回溯算法基础 1.什么是回溯法? 回溯是递归的副产品,只要有递归就会有回溯。 2. 回溯法的效率 因为回溯的本质是穷举,穷举所有可能,然后选出我们想要的答案,如果想让…

【小梦C嘎嘎——启航篇】C++四大类型转换

😎 前言🙌C四大类型转换什么是类型转换C语言中的类型转换为什么C要嫌弃C语言的类型转换?自行搞一套呢?C强制类型转换1、static_cast2、reinterpret_cast3、const_cast4、dynamic_cast为什么要支持向下转呢? RTTI 总结撒…

【Hive】自定义函数从编写到应用的整个流程(以UDF为例)

1. 编写UDF程序 以Java为例,编写一个字符串反转的函数(工程依赖部分略): package com.example;import org.apache.hadoop.hive.ql.exec.UDF; import org.apache.hadoop.hive.ql.exec.Description; import org.apache.hadoop.hiv…