百度360必应搜狗淘宝本站头条
当前位置:网站首页 > IT技术 > 正文

最通俗易懂的命令模式讲解(命令模式百科)

wptr33 2025-06-13 17:20 13 浏览

我们先不讲什么是命令模式,先通过一个场景来引出命令模式,看看命令模式能解决什么样的问题。

现在有一个渣男张三,他有还几个女朋友,你现在是不是还是单身狗,你就说你气不气?然后他需要每天分别叫几个女朋友起床和睡觉(就只是睡觉,别想歪)。那么我们怎么用代码来模拟实现这个事情呢?常规解决办法是我们建立一个女朋友的接口(毕竟女朋友多,得立规矩),然后渣男张三分别聚合多个女朋友,分别调用每个女朋友的起床和睡觉。类图大致如下:


类图不规范,凑合看吧,大致能理解想表达的意思就行了,不要在意这些细节。

这么设计有很多问题,首先就是张三和各个女朋友都是聚合关系,太亲密,不利于扩展,比如张三又新交一个女朋友,就得改张三的代码,不符合开闭原则。说白了就是不方便张三扩张业务。那么显然张三也不会同意你这么一通操作断了他的后路的。

那么此时就需要命令模式来一显身手了,命令模式可以将请求发送者和接收者完全解耦,发送者与接收者之间没有直接引用关系,发送请求的对象只需要知道如何发送请求,而不必知道如何完成请求

有点抽象,不过没关系,看不懂就不看。我们直接来看看类图。


这里的Command(命令)就是张三可以执行的操作,这里是一个接口或者抽象类。

Receiver(命令接收者)就是张三的女朋友(复数加s)。

Invoker(调用者)将两者整合到一起,负责调用命令对象执行请求。

GetUpCommand和SleepCommand是具体的命令操作。

根据类图写出来的代码如下:

public interface Command {

void execute();

void undo();

}

/** * 用作初始化 *

/public class NullCommand implements Command {

public void execute() {

}

public void undo() {

}

}

/**

* 命令接收者接口

**/

public interface Receiver {

public void getUp();

public void sleep();

}

/** * 女朋友1号小红 */

public class XiaoHong implements Receiver {

public void getUp(){

System.out.println("小红起来了");

}

public void sleep(){

System.out.println("小红睡下了");

}

}

public class XiaoHongGetUpCommand implements Command {

public XiaoHong xiaoHong;

public XiaoHongGetUpCommand(XiaoHong xiaoHong){

this.xiaoHong = xiaoHong;

}

public void execute() {

this.xiaoHong.getUp(); //小红起床

}

public void undo() {

this.xiaoHong.sleep(); //撤销,小红睡觉

}

}

public class XiaoHongSleepCommand implements Command {

public XiaoHong xiaoHong;

public XiaoHongSleepCommand(XiaoHong xiaoHong){

this.xiaoHong = xiaoHong;

}

public void execute() {

this.xiaoHong.sleep(); //小红睡觉

}

public void undo() {

this.xiaoHong.getUp(); //撤销,小红起床

}

}

public class Invoker {

private Command[] getUpCommand = new Command[5]; // 此处给渣男张三预留5个女朋友的位置 private Command[] sleepCommand = new Command[5]; // 同上 public Command undoCommand; //记录上一次操作,用作撤回命令 public Invoker(){

for (int i =0;i < 5;i++){

getUpCommand[i] = new NullCommand();

sleepCommand[i] = new NullCommand();

}

}

public void setCommand(int index, Command getUpCommand, Command sleepCommand){

this.getUpCommand[index] = getUpCommand;

this.sleepCommand[index] = sleepCommand;

}

public void setDown(int index){

undoCommand = getUpCommand[index];

getUpCommand[index].execute();

}

public void setUp(int index){

undoCommand = sleepCommand[index];

sleepCommand[index].execute();

}

/** * 撤销操作,起床的重新睡觉,睡觉的就起床 */ public void undo() {

undoCommand.undo();

}

}

public class Demo {

public static void main(String[] args) {

XiaoHong xiaoHong = new XiaoHong();

XiaoHongSleepCommand xiaoHongSleepCommand = new XiaoHongSleepCommand(xiaoHong);

XiaoHongGetUpCommand xiaoHongGetUpCommand = new XiaoHongGetUpCommand(xiaoHong);

invoker.setCommand(1, xiaoHongSleepCommand, xiaoHongGetUpCommand);

invoker.setUp(1);

invoker.setDown(1);

invoker.undo();

}

}

用命令模式就比较完美的解决张三的问题,而且还方便了张三继续找女朋友,他一定会感谢你的。如果新交了女朋友。比如来第二个女朋友小花,扩展方法如下:

/** * 女朋友2号小花 */

public class XiaoHua implements Receiver{

public void getUp(){

System.out.println("小花起来了");

}

public void sleep(){

System.out.println("小花躺下了");

}

}

public class XiaoHuaGetUpCommand implements Command {

public XiaoHua xiaoHua;

public XiaoHuaGetUpCommand(XiaoHua xiaoHua){

this.xiaoHua = xiaoHua;

}

public void execute() {

this.xiaoHua.getUp();

}

public void undo() {

this.xiaoHua.sleep();

}

}

public class XiaoHuaSleepCommand implements Command {

public XiaoHua xiaoHua;

public XiaoHuaSleepCommand(XiaoHua xiaoHua){

this.xiaoHua = xiaoHua;

}

public void execute() {

this.xiaoHua.sleep();

}

public void undo() {

this.xiaoHua.getUp();

}

}

public class Demo {

public static void main(String[] args) {

XiaoHong xiaoHong = new XiaoHong();

XiaoHongSleepCommand xiaoHongSleepCommand = new XiaoHongSleepCommand(xiaoHong);

XiaoHongGetUpCommand xiaoHongGetUpCommand = new XiaoHongGetUpCommand(xiaoHong);

invoker.setCommand(1, xiaoHongSleepCommand, xiaoHongGetUpCommand);

invoker.setUp(1);

invoker.setDown(1);

invoker.undo();

System.out.println("-----------------------------------");

XiaoHua xiaoHua = new XiaoHua();

XiaoHuaSleepCommand xiaoHuaSleepCommand = new XiaoHuaSleepCommand(xiaoHua);

XiaoHuaGetUpCommand xiaoHuaGetUpCommand = new XiaoHuaGetUpCommand(xiaoHua);

Invoker invoker = new Invoker();

invoker.setCommand(0, xiaoHuaSleepCommand, xiaoHuaGetUpCommand);

invoker.setUp(0);

invoker.setDown(0);

invoker.undo();

}

}

怎么样,是不是很方便!再回上面看看命令模式的定义,是不是有点明白?

相关推荐

redis的八种使用场景

前言:redis是我们工作开发中,经常要打交道的,下面对redis的使用场景做总结介绍也是对redis举报的功能做梳理。缓存Redis最常见的用途是作为缓存,用于加速应用程序的响应速度。...

基于Redis的3种分布式ID生成策略

在分布式系统设计中,全局唯一ID是一个基础而关键的组件。随着业务规模扩大和系统架构向微服务演进,传统的单机自增ID已无法满足需求。高并发、高可用的分布式ID生成方案成为构建可靠分布式系统的必要条件。R...

基于OpenWrt系统路由器的模式切换与网页设计

摘要:目前商用WiFi路由器已应用到多个领域,商家通过给用户提供一个稳定免费WiFi热点达到吸引客户、提升服务的目标。传统路由器自带的Luci界面提供了工厂模式的Web界面,用户可通过该界面配置路...

这篇文章教你看明白 nginx-ingress 控制器

主机nginx一般nginx做主机反向代理(网关)有以下配置...

如何用redis实现注册中心

一句话总结使用Redis实现注册中心:服务注册...

爱可可老师24小时热门分享(2020.5.10)

No1.看自己以前写的代码是种什么体验?No2.DooM-chip!国外网友SylvainLefebvre自制的无CPU、无操作码、无指令计数器...No3.我认为CS学位可以更好,如...

Apportable:拯救程序员,IOS一秒变安卓

摘要:还在为了跨平台使用cocos2d-x吗,拯救objc程序员的奇葩来了,ApportableSDK:FreeAndroidsupportforcocos2d-iPhone。App...

JAVA实现超买超卖方案汇总,那个最适合你,一篇文章彻底讲透

以下是几种Java实现超买超卖问题的核心解决方案及代码示例,针对高并发场景下的库存扣减问题:方案一:Redis原子操作+Lua脚本(推荐)//使用Redis+Lua保证原子性publicbo...

3月26日更新 快速施法自动施法可独立设置

2016年3月26日DOTA2有一个79.6MB的更新主要是针对自动施法和快速施法的调整本来内容不多不少朋友都有自动施法和快速施法的困扰英文更新日志一些视觉BUG修复就不翻译了主要翻译自动施...

Redis 是如何提供服务的

在刚刚接触Redis的时候,最想要知道的是一个’setnameJhon’命令到达Redis服务器的时候,它是如何返回’OK’的?里面命令处理的流程如何,具体细节怎么样?你一定有问过自己...

lua _G、_VERSION使用

到这里我们已经把lua基础库中的函数介绍完了,除了函数外基础库中还有两个常量,一个是_G,另一个是_VERSION。_G是基础库本身,指向自己,这个变量很有意思,可以无限引用自己,最后得到的还是自己,...

China&#39;s top diplomat to chair third China-Pacific Island countries foreign ministers&#39; meeting

BEIJING,May21(Xinhua)--ChineseForeignMinisterWangYi,alsoamemberofthePoliticalBureau...

移动工作交流工具Lua推出Insights数据分析产品

Lua是一个适用于各种职业人士的移动交流平台,它在今天推出了一项叫做Insights的全新功能。Insights是一个数据平台,客户可以在上面实时看到员工之间的交流情况,并分析这些情况对公司发展的影响...

Redis 7新武器:用Redis Stack实现向量搜索的极限压测

当传统关系型数据库还在为向量相似度搜索的性能挣扎时,Redis7的RedisStack...

Nginx/OpenResty详解,Nginx Lua编程,重定向与内部子请求

重定向与内部子请求Nginx的rewrite指令不仅可以在Nginx内部的server、location之间进行跳转,还可以进行外部链接的重定向。通过ngx_lua模块的Lua函数除了能实现Nginx...