- Java锁的逻辑(结合对象头和ObjectMonitor)
- 还在用饼状图?来瞧瞧这些炫酷的百分比可视化新图形(附代码实现)⛵
- 自动注册实体类到EntityFrameworkCore上下文,并适配ABP及ABPVNext
- 基于Sklearn机器学习代码实战
命令模式(Command Pattern)是一种行为型设计模式,旨在将请求发送者和接收者解耦,将一个请求封装为一个对象,从而允许您参数化客户端对象以进行不同的请求、排队请求或记录请求,并支持可撤销操作.
命令模式的核心思想是将一个请求包装成一个对象,包括请求的参数和接收者对象,然后客户端只需要调用该对象的方法来执行请求,而不需要关心请求的具体细节。这种方式使得请求的发送者和接收者之间的关系变得松耦合,同时支持一些附加功能,如命令的撤销和重做.
为了更好地理解命令模式,让我们考虑一个实际的例子:遥控器控制家电。假设您有一个遥控器,可以控制不同种类的家电设备,如电视、音响和灯。每个按钮都代表一个命令,例如打开电视、关闭音响、调暗灯光等。命令模式可以用于实现这种遥控器系统.
在这个例子中,每个命令(如打开电视)都被封装成一个命令对象,包括具体的操作(执行命令)和接收者对象(执行命令的设备)。遥控器上的按钮只需要关联一个命令对象,并在按下按钮时执行该命令。这种方式使得遥控器可以轻松控制不同种类的家电设备,而不需要关心它们的具体实现.
命令模式的结构包括以下几个关键部分:
Command(命令):定义一个执行操作的接口,通常包括一个 execute 方法,负责执行具体的命令.
ConcreteCommand(具体命令):实现命令接口,将一个接收者对象绑定到一个操作。它负责调用接收者的方法来执行命令.
Receiver(接收者):负责执行与请求相关的具体操作,是命令真正的执行者.
Invoker(调用者):负责将命令对象传递给接收者执行命令,它不需要知道命令的具体细节,只需调用命令的 execute 方法即可.
Client(客户端):创建具体命令对象,并将命令对象与接收者对象关联,然后将命令对象传递给调用者来执行.
实现命令模式时,通常遵循以下步骤:
定义命令接口(Command),其中包括一个 execute 方法用于执行命令.
创建具体命令类(ConcreteCommand),实现命令接口,并在构造函数中绑定一个接收者对象.
定义接收者类(Receiver),负责执行具体的操作.
创建调用者类(Invoker),负责接收命令对象并执行命令.
在客户端中创建命令对象和接收者对象,并将它们关联。然后将命令对象传递给调用者,由调用者执行命令.
现在,让我们通过 Java 代码来实现上述遥控器控制家电的命令模式.
// 1. 定义命令接口
interface Command {
void execute();
}
// 2. 创建具体命令类
class LightOnCommand implements Command {
private Light light;
public LightOnCommand(Light light) {
this.light = light;
}
@Override
public void execute() {
light.turnOn();
}
}
class LightOffCommand implements Command {
private Light light;
public LightOffCommand(Light light) {
this.light = light;
}
@Override
public void execute() {
light.turnOff();
}
}
// 3. 定义接收者类
class Light {
public void turnOn() {
System.out.println("Light is on");
}
public void turnOff() {
System.out.println("Light is off");
}
}
// 4. 创建调用者类
class RemoteControl {
private Command command;
public void setCommand(Command command) {
this.command = command;
}
public void pressButton() {
command.execute();
}
}
// 5. 客户端代码
public class Client {
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();
}
}
命令模式在实际应用中有多种典型场景,包括但不限于:
遥控器控制:如上面的示例所示,可以用命令模式实现遥控器来控制不同的家电设备,如电视、音响和灯.
文本编辑器操作:文本编辑器中的撤销、重做、剪切、复制、粘贴等操作可以使用命令模式来实现.
菜单系统:图形用户界面(GUI)应用中的菜单项和按钮操作可以通过命令模式来处理.
游戏中的动作:在游戏中,角色的动作和命令(如攻击、防御、跳跃等)可以使用命令模式来处理.
多级撤销操作:命令模式支持撤销和重做操作,因此在需要多级撤销的应用中很有用,如图像编辑器或CAD软件.
日程安排应用:在日程安排应用中,可以使用命令模式来处理添加、编辑、删除事件等操作.
优点:
解耦发送者和接收者:命令模式将请求的发送者和接收者解耦,发送者不需要知道接收者的具体实现,从而降低了系统的耦合度.
支持撤销和重做:命令模式可以轻松支持命令的撤销和重做,因为每个命令对象都包含了执行和撤销操作的逻辑.
支持扩展:可以轻松添加新的命令和接收者,而无需修改现有的客户端代码.
支持排队请求:命令模式允许将请求排队,以便按照先后顺序执行,或者实现任务调度.
缺点:
可能导致命令类膨胀:如果有大量的命令操作,可能会导致命令类的膨胀,增加维护的复杂性.
不适用于所有情况:命令模式不适用于所有场景,特别是对于简单的命令,直接调用方法可能更加简单和高效.
增加代码复杂性:引入命令模式会增加一定的代码复杂性,因为需要创建额外的命令类和接收者类.
策略模式(Strategy Pattern):策略模式也可以用于封装可互换的行为,但它的主要目的是在运行时选择算法或策略,而不是将请求封装成命令对象.
观察者模式(Observer Pattern):观察者模式用于定义对象之间的一对多依赖关系,当一个对象状态发生变化时,所有依赖于它的对象都会收到通知。与命令模式不同,观察者模式不涉及命令的封装和执行.
中介者模式(Mediator Pattern):中介者模式用于减少对象之间的直接耦合,将对象之间的通信集中在一个中介者对象中。与命令模式不同,中介者模式通常用于管理对象之间的交互,而不是将请求封装成命令.
命令模式是一种有用的设计模式,它允许将请求封装成命令对象,从而解耦请求的发送者和接收者,支持撤销和重做操作,并提供扩展性和灵活性。命令模式在实际应用中有多种典型场景,包括遥控器控制、文本编辑器操作、菜单系统等。虽然命令模式增加了一些复杂性,但它可以提高代码的可维护性和可扩展性,是面向对象设计中的重要模式之一.
最后此篇关于软件设计模式系列之十六——命令模式的文章就讲到这里了,如果你想了解更多关于软件设计模式系列之十六——命令模式的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
关闭。这个问题是off-topic .它目前不接受答案。 想改进这个问题吗? Update the question所以它是on-topic用于堆栈溢出。 关闭 9 年前。 Improve this
我有一系列 SQL 命令,我想在大约 40 个不同的表上运行。必须有一种方法可以在不编写 40 条不同命令的情况下执行此操作... 我在 SQL Server 中运行它。所有表都有不同的名称,我要操作
我习惯在 PHP 中使用命令“mysql_insert_id()”来返回插入到我的数据库中的最后一行的 id。 在 C# 中的 SQLite 中是否有等效的命令? 谢谢! -阿德娜 最佳答案 选择 l
试图找出一种方法来回填 ds 分区 Hive 表的分区。 我知道如何从 CLI 运行 Hive 命令,例如 $HIVE_HOME/bin/hive -e 'select a.col from tab1
我有 .bat 文件。看起来像下一个 ....many commands1 ftp -i -s:copy.txt ...many commands2 copy.txt 包含下一个命令 open ...
基本上我想输入 show 并检查是否有 show 命令或别名已定义并触发它,如果未定义则触发 git show 。 例如 rm 应该执行 rm 但 checkout 应该执行 git checkout
我公司的主数据库是 iSeries 机器,我已经非常习惯使用 DB2 命令和结构。我现在正在尝试做一个小项目,更新一个包含超过 300 万条记录的表。我想出一种比较和“清理”数据的更快方法是使用 My
我想在带有 Node 的终端中制作一个简单的按钮板,并“blessed”用于连接或运行不同的命令。 ----------------------------------------------- _
我们有一个 selenium IDE 脚本,正在转换为 python webdriver。以下命令未转换: [openWindow | http://mywebsite.com/index.php |
我正在学习这个关于从 GIT HUB 下载和安装 Web 文件的在线教程。我进入主题:启动我们的静态网站,系统提示我输入命令以下载和安装 Web 文件。但是,当我输入命令 yarn install 时
我在 shell 脚本中使用 elif 命令时遇到问题,就像在 fortran 中一样。 我有 100 家公司的员工名单。我想屏蔽那些员工少于 500 人的公司。我的脚本是 rm -f categor
我有一些 Linux 命令可以生成 token 。我在 Linux 机器上使用操作系统库形式的 Python 自动化了这些命令。它工作正常。 但是,当我在 Windows 中尝试相同的代码时,它没有返
本文分享自华为云社区《Git你有可能不知道交互式暂存》,作者:龙哥手记。 本节中的几个交互式 Git 命令可以帮助你将文件的特定部分组合成提交。 当你在修改了大量文件后,希望这些改动能拆分为若干提交而
我想知道如何使用 IN 比较语法来做到这一点。 当前的 SQL 查询是: select * from employee where (employeeName = 'AJAY' and month(e
我在这个位置安装了 Hadoop /usr/local/hadoop$ 现在我想列出 dfs 中的文件。我使用的命令是: hduser@ubuntu:/usr/local/hadoop$ bin/ha
是否有一个单一的 docker 命令可用于清除所有内容?如果正在运行,请停止所有容器、删除所有图像、删除所有卷...等。 最佳答案 我认为没有一个命令可以做到这一点。您首先需要停止所有容器使用 $ d
我基本上是在 clojure/nrepl 模式中寻找与 C-u C-x C-e 或 C-c C-p 等效的 Scheme。 我想要一个 C-x C-e 将输出打印到缓冲区,而不是仅仅在 repl 中。
我可以在 vim 中使用 pudb(一个 ncurses Python 调试器),因为,例如,:!python %在实际的终端窗口中运行。我更喜欢使用 gvim,但 gvim 运行 :!python
我正在尝试编写一个 FFMPEG 命令: 取为 输入 一个视频 input.mp4 和一个图像 pic.jpg 作为 输出 将 input.mp4 拆分为 20 秒的视频,按顺序重命名;对于每个分割视
我想转储视频每帧的比特率。我正在尝试使用 -vstats 获取此信息命令。当我运行此命令时 - ffmpeg -i input.mp4 -vstats 它显示至少应该定义一个文件。 如果有人能建议我任
我是一名优秀的程序员,十分优秀!