设计模式---命令模式

article/2025/10/20 14:36:51

命令模式

命令模式的定义

​ 命令模式是一个高内聚的模式,其定义为:Encapsulate a request as an object,thereby letting you parameterize clients with different requests,queue or log requests,and support undoable operations.(将一个请求封装成一个对象,从而让你使用不同的请求把客户端参数化,对请求排队或者记录请求日志,可以提供命令的撤销和恢复功能。)

​ 补充:什么是高内聚?

​ 高内聚低耦合,是软件工程中的概念,是判断软件设计好坏的标准,主要用于程序的面向对象的设计,主要看类的内聚性是否高,耦合度是否低。目的是使程序模块的可重用性、移植性大大增强。通常程序结构中各模块的内聚程度越高,模块间的耦合程度就越低。

通用类图

在这里插入图片描述

在该类图中,我们看到三个角色:

● Receive接收者角色

该角色就是干活的角色,命令传递到这里是应该被执行的,具体到我们上面的例子中就是Group的三个实现类。

● Command命令角色

需要执行的所有命令都在这里声明。

● Invoker调用者角色

接收到命令,并执行命令。在例子中,(项目经理)就是这个角色。

命令模式比较简单,但是在项目中非常频繁地使用,因为它的封装性非常好,把请求方(Invoker)和执行方(Receiver)分开了,扩展性也有很好的保障,通用代码比较简单

代码结构

通用Receiver类

public abstract class Receiver {//抽象接收者,定义每个接收者都必须完成的业务public abstract void doSomething();
}

为什么Receiver是一个抽象类?那是因为接收者可以有多个,有多个就需要定义一个所有特性的抽象集合——抽象的接收者

具体的Receiver类

public class ConcreteReciver1 extends Receiver{    //每个接收者都必须处理一定的业务逻辑public void doSomething(){}
}
public class ConcreteReciver2 extends Receiver{ //每个接收者都必须处理一定的业务逻辑public void doSomething(){}
}

接收者可以是N个,这要依赖业务的具体定义。命令角色是命令模式的核心

抽象的Command类

public abstract class Command {//每个命令类都必须有一个执行命令的方法public abstract void execute();
}

根据环境的需求,具体的命令类也可以有N个。

具体的Command类

public class ConcreteCommand1 extends Command {//对哪个Receiver类进行命令处理private Receiver receiver; //构造函数传递接收者public ConcreteCommand1(Receiver _receiver){this.receiver = _receiver;}//必须实现一个命令public void execute() {//业务处理this.receiver.doSomething();}
}
public class ConcreteCommand2 extends Command {//哪个Receiver类进行命令处理private Receiver receiver;//构造函数传递接收者public ConcreteCommand2(Receiver _receiver){this.receiver = _receiver;}//必须实现一个命令public void execute() {//业务处理this.receiver.doSomething();}
}

定义了两个具体的命令类,我们可以在实际应用中扩展该命令类。在每个命令类中,通过构造函数定义了该命令是针对哪一个接收者发出的,定义一个命令接收的主体。调用者非常简单,仅实现命令的传递。

调用者Invoker类

public class Invoker {private Command command;//受气包,接受命令public void setCommand(Command _command){this.command = _command;} //执行命令public void action(){this.command.execute();}
}

调用者就像是一个受气包,不管什么命令,都要接收、执行!那我们来看高层模块如何调用命令模式.

public class Client {public static void main(String[] args) {//首先声明调用者InvokerInvoker invoker = new Invoker();//定义接收者Receiver receiver = new ConcreteReciver1();//定义一个发送给接收者的命令Command command = new ConcreteCommand1(receiver);//把命令交给调用者去执行invoker.setCommand(command);invoker.action();}
}

案例

​ 小女孩茱丽(Julia)有一个盒式录音机,此录音机有播音(Play)、倒带(Rewind)和停止(Stop)功能,录音机的键盘便是请求者(Invoker)角色;茱丽(Julia)是客户端角色,而录音机便是接收者角色。Command类扮演抽象命令角色,而PlayCommand、StopCommand和RewindCommand便是具体命令类。茱丽(Julia)不需要知道播音(play)、倒带(rewind)和停止(stop)功能是怎么具体执行的,这些命令执行的细节全都由键盘(Keypad)具体实施。茱丽(Julia)只需要在键盘上按下相应的键便可以了。

录音机是典型的命令模式。录音机按键把客户端与录音机的操作细节分割开来。
  在这里插入图片描述

接收者角色,由录音机类扮演

public class AudioPlayer {public void play(){System.out.println("播放...");}public void rewind(){System.out.println("倒带...");}public void stop(){System.out.println("停止...");}
}

抽象命令角色类

public interface Command {/*** 执行方法*/public void execute();
}

具体命令角色类

public class PlayCommand implements Command {private AudioPlayer myAudio;public PlayCommand(AudioPlayer audioPlayer){myAudio = audioPlayer;}/*** 执行方法*/@Overridepublic void execute() {myAudio.play();}}public class RewindCommand implements Command {private AudioPlayer myAudio;public RewindCommand(AudioPlayer audioPlayer){myAudio = audioPlayer;}@Overridepublic void execute() {myAudio.rewind();}}public class StopCommand implements Command {private AudioPlayer myAudio;public StopCommand(AudioPlayer audioPlayer){myAudio = audioPlayer;}@Overridepublic void execute() {myAudio.stop();}}

调用者角色,由键盘类扮演

public class Keypad {private Command playCommand;private Command rewindCommand;private Command stopCommand;public void setPlayCommand(Command playCommand) {this.playCommand = playCommand;}public void setRewindCommand(Command rewindCommand) {this.rewindCommand = rewindCommand;}public void setStopCommand(Command stopCommand) {this.stopCommand = stopCommand;}/*** 执行播放方法*/public void play(){playCommand.execute();}/*** 执行倒带方法*/public void rewind(){rewindCommand.execute();}/*** 执行播放方法*/public void stop(){stopCommand.execute();}
}

客户端角色,由茱丽小女孩扮演

public class Julia {public static void main(String[]args){//创建接收者对象AudioPlayer audioPlayer = new AudioPlayer();//创建命令对象Command playCommand = new PlayCommand(audioPlayer);Command rewindCommand = new RewindCommand(audioPlayer);Command stopCommand = new StopCommand(audioPlayer);//创建请求者对象Keypad keypad = new Keypad();keypad.setPlayCommand(playCommand);keypad.setRewindCommand(rewindCommand);keypad.setStopCommand(stopCommand);//测试keypad.play();keypad.rewind();keypad.stop();keypad.play();keypad.stop();}
}

宏命令

​ 所谓宏命令简单点说就是包含多个命令的命令,是一个命令的组合。

设想茱丽的录音机有一个记录功能,可以把一个一个的命令记录下来,再在任何需要的时候重新把这些记录下来的命令一次性执行,这就是所谓的宏命令集功能。因此,茱丽的录音机系统现在有四个键,分别为播音、倒带、停止和宏命令功能。此时系统的设计与前面的设计相比有所增强,主要体现在Julia类现在有了一个新方法,用以操作宏命令键。

在这里插入图片描述

系统需要一个代表宏命令的接口,以定义出具体宏命令所需要的接口。

public interface MacroCommand extends Command {/*** 宏命令聚集的管理方法* 可以添加一个成员命令*/public void add(Command cmd);/*** 宏命令聚集的管理方法* 可以删除一个成员命令*/public void remove(Command cmd);
}

具体的宏命令MacroAudioCommand类负责把个别的命令合成宏命令。

public class MacroAudioCommand implements MacroCommand {private List<Command> commandList = new ArrayList<Command>();/*** 宏命令聚集管理方法*/@Overridepublic void add(Command cmd) {commandList.add(cmd);}/*** 宏命令聚集管理方法*/@Overridepublic void remove(Command cmd) {commandList.remove(cmd);}/*** 执行方法*/@Overridepublic void execute() {for(Command cmd : commandList){cmd.execute();}}}	

客户端类JuliaSister

public class Julia {public static void main(String[]args){//创建接收者对象AudioPlayer audioPlayer = new AudioPlayer();//创建命令对象Command playCommand = new PlayCommand(audioPlayer);Command rewindCommand = new RewindCommand(audioPlayer);Command stopCommand = new StopCommand(audioPlayer);MacroCommand marco = new MacroAudioCommand();marco.add(playCommand);marco.add(rewindCommand);marco.add(stopCommand);marco.execute();}
}

运行结果:
在这里插入图片描述

撤销测试

public class Julia {public static void main(String[]args){//创建接收者对象AudioPlayer audioPlayer = new AudioPlayer();//创建命令对象Command playCommand = new PlayCommand(audioPlayer);Command rewindCommand = new RewindCommand(audioPlayer);Command stopCommand = new StopCommand(audioPlayer);MacroCommand marco = new MacroAudioCommand();marco.add(playCommand);//撤销命令marco.remove(playCommand);marco.add(rewindCommand);marco.add(stopCommand);marco.execute();}
}

运行结果:

在这里插入图片描述

命令模式的应用

命令模式的优点

● 类间解耦

调用者角色与接收者角色之间没有任何依赖关系,调用者实现功能时只需调用Command抽象类的execute方法就可以,不需要了解到底是哪个接收者执行。

● 可扩展性

Command的子类可以非常容易地扩展,而调用者Invoker和高层次的模块Client不产生严重的代码耦合。

命令模式的缺点

命令模式也是有缺点的,请看Command的子类:如果有N个命令,问题就出来了,Command的子类就可不是几个,而是N个,这个类膨胀得非常大,这个就需要读者在项目中慎重考虑使用。

命令模式的使用场景

只要你认为是命令的地方就可以采用命令模式,例如,在GUI开发中,一个按钮的点击是一个命令,可以采用命令模式;模拟DOS命令的时候,当然也要采用命令模式;触发-反馈机制的处理等。

命令模式在 Spring 框架 JdbcTemplate 应用的源码分析

Spring 框架的 JdbcTemplate 就使用到了命令模式

在这里插入图片描述

模式角色分析说明

StatementCallback 接口 ,类似命令接口(Command)

class QueryStatementCallback implements StatementCallback, SqlProvider , 匿名内部类, 实现了命令接口, 同时也充当命令接收者

命令调用者 是 JdbcTemplate , 其中 execute(StatementCallback action) 方法中,调用 action.doInStatement 方法. 不同的 实现 StatementCallback 接口的对象,对应不同的 doInStatemnt 实现逻辑

另外实现 StatementCallback 命令接口的子类还有 ExecuteStatementCallback、StreamStatementCallback、BatchUpdateStatementCallback等。


http://chatgpt.dhexx.cn/article/HtpTYAZl.shtml

相关文章

【设计模式】命令模式 ( 简介 | 适用场景 | 优缺点 | 代码示例 )

文章目录 一、命令模式简介二、命令模式 适用场景三、命令模式 优缺点四、命令模式 与 备忘录模式五、命令模式 代码示例1、命令接口2、发布命令类3、关闭命令类4、游戏类5、命令执行者类6、测试类 一、命令模式简介 命令模式 : 将 不同的请求 封装成 不同的请求对象 , 以便 使…

设计模式(16)命令模式

**定义&#xff1a;**将一个请求封装成一个对象&#xff0c;从而让你使用不同的请求把客户端参数化&#xff0c;对请求排队或者记录请求日志&#xff0c;可以提供命令的撤销和恢复功能。 **类型&#xff1a;**行为类模式 类图&#xff1a; 命令模式的结构 ​ 顾名思义&#…

什么是命令模式?

一、命令模式的定义 命令是对命令的封装&#xff0c;每一个命令都是一个操作&#xff0c;请求方发出请求&#xff0c;接收方接收请求&#xff0c;并执行操作。命令模式解耦了请求方和接收方&#xff0c;命令模式属于行为型模式 二、命令模式的uml图和通用写法 uml 通用写法 …

设计模式之命令模式详解

1 概述 日常生活中&#xff0c;我们出去吃饭都会遇到下面的场景。我们可以将女招待理解成一个请求的发送者&#xff0c;用户通过它来发送一个“点餐”请求&#xff0c;而厨师是“点餐”请求的最终接收者和处理者&#xff0c;在图中&#xff0c;顾客和厨师之间并不存在直接耦合…

命令模式

一、命令模式介绍 在软件设计中&#xff0c;我们经常需要向某些对象发送请求&#xff0c;但是并不知道请求的接收者是谁&#xff0c;也不知道被请求的操作是哪个。我们只需要在程序运行时指定具体的请求接收者即可&#xff0c;此时可以使用命令模式来设计。 命令模式使得请求发…

Java设计模式——命令模式

文章目录 命令模式 命令模式 命令模式很好理解&#xff0c;举个例子&#xff0c;司令员下令让士兵去干件事情&#xff0c;从整个事情的角度来考虑&#xff0c;司令员的作用是&#xff0c;发出口令&#xff0c;口令经过传递&#xff0c;传到了士兵耳朵里&#xff0c;士兵去执行…

如何设置IPv4和IPv6报文的DSCP值——网络测试仪实操

一、操作说明 在QoS测试中&#xff0c;经常要设置不同优先级的报文&#xff0c;来验证被测设备对于优先级的调度。所以&#xff0c;我们就要了解如何设置IPv6和IPv6报文中的DSCP&#xff08;大部分使用DSCP值&#xff0c;也会用到TOS值&#xff09; 这里我们使用测试接交换机&…

DSCP vs IPv4 Tos

首先看IPv4包头如下 其中&#xff0c;Qos用到的是Tos定义有下面两种&#xff1a; 老的IPv4 TOS Byte定义和值 新的DSCP定义和值 DSCP值 DSCP ValueMeaningDrop ProbabilityEquivalent IP Precedence Value101 110 (46)High Priority Expedited Forwarding (EF)N/A101 – …

c语言socket设置IPV4/6的dscp值

环境&#xff1a;linux centos7 、x86 、UDP包 使用sock需要增加头文件 #include <sys/socket.h> #include <sys/types.h> 设置方法很简单&#xff0c;都是使用setsockopt函数&#xff0c;就是找资料及如何太麻烦&#xff0c;尤其是IPV6。需要注意IPV4设置的是I…

tos cos dscp 区别和作用

tos cos 和dscp 都是通过iptable 的mange 的mark 标签来更改的。 谈到qos首先需要了解qos调度的几个重要过程,qos调度过程包括网络入口数据流量的分类和标记、骨干网设备上的拥塞避免和拥塞管理、网路出口的队列调度这几个重要过程. 1、cos和tos的区别: 通过acl对流量进行分类以…

IP优先级和DSCP之间的关系

1. IP优先级和DSCP之间的关系 DiffServ体系定义的DS字段&#xff0c;取代IPv4中ToS字段作出有关数据包分类和流量调节功能的策略。 1.1. ToS字段 在IPv4的报文头中&#xff0c;TOS字段是1字节&#xff0c;根据RFC1122的定义&#xff0c;IP优先级&#xff08;IPPrecedence&…

802.1P优先级、IP优先级、TOS优先级及DSCP优先级的分类和对应

1、802.1P优先级&#xff08;有时也称COS优先级&#xff09;&#xff1a; 802.1p用户优先级定义在二层802.1Q 标签头中的TCI字段中。&#xff0c;和VLAN ID一起使用&#xff0c;位于高位起16-18bit字段&#xff0c;长度3bit&#xff0c;取值范围0-7&#xff0c;0优先级最低&…

DSCP 与IP 优先级IP优先级

首先看IPv4包头如下 其中&#xff0c;Qos用到的是Tos定义有下面两种&#xff1a; 老的IPv4 TOS Byte定义和值 新的DSCP定义和值 DSCP值 DSCP Value Meaning Drop Probability Equivalent IP Precedence Value 101 110 (46) High Priority Expedited Forwarding (EF) N/A…

IP Precedence、DSCP、TOS

刚开始接触QoS时&#xff0c;经常会被IP Precedence、DSCP、TOS这些名词搞迷糊&#xff0c;那么接下来就梳理一下。 首先 IP Precedence IPv4中有8bit作为TOS字段&#xff0c;一开始RFC791定义了TOS前三位为IP Precedence&#xff0c;划分了8个优先级&#xff0c;可用于流分类…

【网络】Cos和ToS和DSCP|Qos|PHB的含义和区别以及映射

目录 视频教程&#xff1a; 介绍和区别 Qos/Cos IP-TOS&#xff08;IPP/CS&#xff09;和DSCP PHB&#xff08;Per-Hop-Behaviors&#xff09; 区别 各个等级的DSCP值和含义(PHB) 映射 COS到DSCP的映射 IP-Precedence到DSCP的映射&#xff08;Tos-->DSCP&#xff09…

TOS 和DSCP

IPv4报文中有三种承载QoS优先级标签的方式&#xff0c;分别为基于二层的CoS字段&#xff08;IEEE802.1p&#xff09;的优先级、基于IP层的IP优先级&#xff08;IP Precedence&#xff09;字段ToS优先级和基于IP层的DSCP&#xff08;Differentiated Services Codepoint&#xff…

什么是DSCP,如何使用DSCP标记搭配ROS策略

什么是DSCP&#xff0c;如何使用DSCP标记搭配ROS策略 一、什么是DSCP DSCP&#xff1a;差分服务代码点&#xff08;Differentiated Services Code Point&#xff09;&#xff0c;IETF于1998年12月发布了Diff-Serv&#xff08;Differentiated Service&#xff09;的QoS分类标准…

TOS 和 DSCP理解

背景 IPv4报文中有三种承载QoS优先级标签的方式&#xff0c;分别为基于二层的CoS字段&#xff08;IEEE802.1p&#xff09;的优先级、基于IP层的IP优先级&#xff08;IP Precedence&#xff09;字段ToS优先级和基于IP层的DSCP&#xff08;Differentiated Services Codepoint&am…

谈谈ES5和ES6的区别

我们都知道JavaScript是由三部分组成&#xff1a; 1. ECMAScript(核心)&#xff1a;规定了语言的组成部分>语法、类型、语句、关键字、保留字、操作符、对象 2. BOM(浏览器对象模型): 支持访问和操作浏览器窗口&#xff0c;可以控制浏览器显示页面以外的部分。 3. DOM(文…

ES5基础语法

一.类与对象 class father {that this;constructor(uname, age) {this.uname uname;this.age age;}sing(song) {console.log(this.uname song);}}class son extends father {constructor(uname,age) {super(uname,age);this.unameuname;this.age age;}sing(song){console.…