设计模式第6式:命令模式

news2024/12/23 11:07:35

前言

命令模式关注这样一种场景:指令发布者无需关注指令是怎么执行的,只需要指定具体的执行者,具体的指令由执行者来完成。命令模式将指令发布动作和指令执行动作解耦。

我在刚开始学习命令模式的时候,比较困惑它的使用场景。它不像之前讲的工厂模式,观察者模式那样有明确的使用场景。其实可以将命令模式应用在这个场景就比较好理解了:软件项目经理不可能完成所有的开发任务,他将软件开发的不同阶段设置为不同的指令,比如软件设计指令、软件开发指令、软件测试指令,然后将不同的指令指定不同的人员。

正文

1、餐厅点单场景

我们以面向对象的思维来看真实的餐厅点单场景是什么样的:
1、顾客创建一个订单(createorder());
2、招待者拿走订单(takeOrder());放在后厨柜台,然后通知厨师备餐(orderUp());
3、厨师备餐出餐(doSomething(),output);

我们来分析其中的角色:
1、把订单想象成封装了备餐请求的对象,对象是可以传递的。订单接口只包含一个方法orderUp(),这个方法封装备餐需要的所有动作。订单实现类内有一个厨师的引用。这些都被封装起来,所以招待者不需要知道订单上有什么,也不需要谁来备餐。
2、招待者的工作只是接受订单,然后调用订单的orderUp方法。
3、厨师具有备餐的技能。厨师和招待者是完全解耦的。

将以上场景进行抽象成命令模式,那这个模式的核心就是将“发出请求的对象”和“接受与执行请求的对象”解耦。

2、将餐厅抽象成命令模式

我们重新来讲命令模式:
在这里插入图片描述
三个角色:
1、调用者:发起命令的角色;
2、命令:封装执行者及其方法;
3、执行者:真正的动作实施人;

3、实现命令模式

上面讲了一堆,我们来实际写一个命令模式。首先,我们需要一个命令类接口,接口很简单,只有一个执行方法。

public interface Command {
	void execute();
}

然后,实现一个打开电灯的具体命令类。

public class LightOnCommand implements Command {
	Light light; // 命令类持有接受者的引用

	public LightOnCommand(Light light) {
		this.light = light;
	}
	 
	@Override
	public void execute() {  // 封装了接受者的一系列方法
		light.on(); // 这里可以编排多个方法
	}
}

命令对象持有了一个接受者,它是实际干活的。

public class Light {
	public void on() {
		sout("打开灯");
	}
}

有了命令类后,我们来看怎么使用它。我们来创建一个调用者。

public class Controller {
	Command command; // 持有一个命令对象

	public void setCommand(Command command) { // 可设置具体命令对象
		this.command = command;
	}

	public void pressButton() { // 调用者按下按钮就能发起命令
		command.execute(); // 调用命令对象封装好的方法
	}
}

最后,我们来验证一下调用者怎么执行。

public class Test {
	main() {
		Controller contr = new Controller();
		
		Light light = new Light();
		LightCommand lightcomm = new LightCommand(light); //命令封装了接受者和一系列实现动作

		contr.setCommand(lightcomm); // 调用者配置命令
		contr.pressButton(); // 调用者执行方法
	}
}

4、怎么理解“命令”

通过上面这个简单的例子,已经完整展示了命令模式的全貌。如果没有“命令”这个类,那么上面这个例子将是什么样的呢?那么调用者将要直接面对千百个功能复杂的接受者,调用者将要创建出多个接受者,并排列组合一系列方法。这会导致调用者类非常复杂,难以维护,后续改变功能或新增功能都要改动调用者。

现在使用命令模式后,将调用者和接受者解耦开,调用者只关注要给谁下命令,由具体的命令对象来编排众多调用者及其方法。

那么怎么理解“命令”呢?万物皆对象,“命令”首先是一个对象,它做了什么呢?它的重点工作就是编排功能,它可以持有真正执行动作的对象,然后在execute()方法中编排方法。

上面的案例中,调用者按下按钮就会打开灯,如果我想改变按钮的功能呢?比如改成关灯,或者先关灯再拉窗帘。这种情况下,调用者Controller的代码是不用改动的,我们要做的是新增命令类,然后在命令类的execute()方法中改成调用执行者的关灯方法,或者先调关灯方法再调拉窗帘方法。这样就真正做到了对修改关闭,对扩展开放。

5、命令模式在Java中的应用

Java中的线程类Thread就是使用命令模式的典型案例。线程启动的方法是Thread.start(),线程具体做什么事Thread类不关心,是在Runnable接口的run()方法中实现的。

类比上面讲的例子,Thread类就是调用者,它只管调用start()方法来启动线程,线程做什么由命令来执行;Runnable接口就是命令类Command,里面的run()方法就是命令类的execute()方法,具体执行什么由Runnable的实现类进行实现。这样一来,Thread类永远不用变化,试想一下如果Thread类需要改变,它怎么作为JDK中的工具类供大家使用。

总结

命令模式常用于解耦动作发起者和动作执行者,并提供了编排动作的功能。

我一开始学习设计模式的时候,容易分不清“客户端”是谁。比如命令模式中我会认为调用者Controller是“客户端”,观察者模式中的主题Subject是“客户端”。之所以会这样认为,是因为它们中有一个触发动作的方法,比如Controller中按按钮动作,主题Subject中发布事件的方法。其实它们是作为各自模式的一部分,只是它们作为门面直接面向了客户端,真正的客户端应该是调用它们的类。有没有发现,很多设计模式的目标就是保持这个门面代码不用修改,去扩展门面后面的类。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【JavaGuide面试总结】高性能篇

【JavaGuide面试总结】高性能篇1.静态资源是如何被缓存到 CDN 节点中的?2.如何找到最合适的 CDN 节点?3.如何防止资源被盗刷?1.静态资源是如何被缓存到 CDN 节点中的? 你可以通过预热的方式将源站的资源同步到 CDN 的节点中。这样…

docker安装达梦8,并且远程连接数据库

官方技术文档地址:https://eco.dameng.com/document/dm/zh-cn/start/index.html 不能直接拉取达梦数据库的镜像,需要去官方提供的下载地址下载镜像,上传到服务器 下载地址:https://eco.dameng.com/download/ 下载完成之后确认下…

【黑马SpringCloud(3)】Docker镜像容器操作Docker-Compose的使用

Docker初识DockerDocker与虚拟机的区别Docker架构DockerHubDocker镜像操作Docker容器操作数据卷自定义镜像镜像结构Dockerfile构建Java项目基于java:8-alpine镜像,构建java项目集群部署微服务Docker-Compose镜像仓库搭建私有镜像仓库Docker安装:https://…

Windows Server 2008 系统加固(2)

实验简介 实验所属系列: 系统安全 实验对象: 本科/专科信息安全专业 相关课程及专业: 服务器安全 实验时数(学分):2学时 实验类别: 实践实验类预备知识 Windows server 2008 是微软公司的…

4 -【Faster R-CNN】之 RPN Head 代码精读

【Faster R-CNN】之 RPN Head 代码精读1、前言2、RPN Head 网络结构3、代码4、相关问题1、前言 在上一篇文章 【Faster R-CNN】之 backbone 代码精读 中,我们学习了创建 backbone,并将 batch 中的图片通过backbone 获得了 feature maps。 batch 的 feat…

linux的shell命令中grep命令返回值问题

linux的shell命令中grep命令返回值问题如何查看上一个命令退出状态?$?的作用set -e的作用一个代码例子shell脚本中的错误示例如何查看上一个命令退出状态? $?的作用 $? 是一个特殊变量,用来获取上一个命令的退出状态,或者上一…

【C语言练习】杨氏矩阵、杨辉三角

目录一:杨氏矩阵🐻何为杨氏矩阵?🐻题目描述:🐻思路一:🐻思路二:二:杨辉三角🐻何为杨辉三角?🐻题目描述:&#…

python数据结构(四):dict(字典)

一、定义字典 1、使用花括号,可以直接定义字典 使用一对大括号{},数据是成对存在的,每一行冒号左边的是键(key),右边的是值(value),称作键值对,以逗号分隔开。键是不能重…

Fennec:针对类Unix操作系统的多功能事件应急响应工具箱

关于Fennec Fennec是一个针对类Unix操作系统的多功能事件应急响应工具箱,Fennec基于Rust开发,可以帮助广大研究人员在类Unix操作系统上实现网络安全事件应急响应。除此之外,Fennec还支持广大研究人员自行开发相关的配置文件,并增…

第一章.感知机

第一章.感知机 1.感知机的简介 1).简介 感知机接收多个输入信号,输出一个信号。感知机的信号只有"流/不流"(1/0)两种取值[0:对应不传递信号,1:对应传递信号] 2).感知机图像描述的两种方式 ①.第一种方式: …

css3 transform(变形) 实战示例

1--利用transform实现居中效果 <div class"center">.... </div> .center{text-align: center;background-color: #fff;border-radius: 20px;width: 300px;height: 350px;position: absolute;left: 50%;top: 50%;transform: translate(-50%,-50%); } 让…

第一章 隐私计算科普解读

前言 提醒&#xff1a;全文10千字&#xff0c;预计阅读时长15分钟&#xff1b;读者&#xff1a;对隐私计算感兴趣的小伙伴&#xff1b;目的&#xff1a;读者利用15~30 分钟对本文沉浸式阅读理解&#xff0c;能够掌握隐私计算 80% 的概念&#xff1b;关键词 &#xff1a;隐私计算…

全网最细------爬取4k付费高清大图(免费下载再也不是梦)

本次案例将教大家免费爬取4k高清付费大图&#xff0c;即使你是爬虫新手&#xff0c;也可以食用本次文章实现你的免费下载梦,话不多说&#xff0c;先看效果 网站视图: 看到这些图片你是否怦然心动&#xff0c;跟着我一起看下去. 下图是图片下载后保存的目录 一.思路分析 首…

QEMU安装Windows 10的完整过程

零、环境介绍 宿主机&#xff1a; Ubuntu 22.04.1 LTS Windows 10镜像&#xff1a; Windows10_20H2.iso QEMU版本&#xff1a; qemu-img version 7.1.0 Copyright (c) 2003-2022 Fabrice Bellard and the QEMU Project developers 一、安装过程 1. 创建镜像文件 使用qe…

7、标识重复个案

目录 一、导入或打开数据 二、选择“数据”菜单下的“标识重复个案”&#xff0c;弹出相应对话框 三、在上个步骤的基础上&#xff0c;将左边的所有变量都移至右边“定义匹配个案的依据”中&#xff0c;其他的选项没有特殊要求的可保持默认&#xff0c;单击“确定”即完成。…

Cookie 和 Session 详解 及实现用户登陆功能

Cookie是啥? 浏览器提供的在客户端存储数据的一种机制。(由于浏览器禁止了网页中的代码直接访问磁盘的文件因此要想再网页中实现数据的持久化存储、就可以使用Cookie这样的机制 Cookie 里面存什么? 键值对结构.键和值都是程序猿自定义的~Cookie从哪里来? 服务器返回响应…

【Java|golang】2325. 解密消息

给你字符串 key 和 message &#xff0c;分别表示一个加密密钥和一段加密消息。解密 message 的步骤如下&#xff1a; 使用 key 中 26 个英文小写字母第一次出现的顺序作为替换表中的字母 顺序 。 将替换表与普通英文字母表对齐&#xff0c;形成对照表。 按照对照表 替换 mess…

【23 Java后端】小米

一、JavaSE JaveEE —— Java 数据结构 —— 集合类 1. HashMap 底层&#xff0c;链表与红黑树转换原因 JDK 1.7 HashMap 底层使用 “数组链表” 实现&#xff0c;数组为主体&#xff0c;链表为了解决 哈希冲突JDK 1.8 HashMap 底层使用 “数组链表红黑树” 实现 当链表长度…

bootstrap是什么,bootstrap如何使用

bootstrap是什么 Bootstrap 是最受欢迎的 HTML、CSS 和 JS 框架&#xff0c;用于开发响应式布局、移动设备优先的 WEB 项目。 使用bootstrap 下载地址

Power BI散点图

散点图可以让一大堆令人困惑的散乱数据变得通俗易懂&#xff0c;并能让你从这些庞杂数据中发现一些表面上看不到的关系&#xff0c;数据量越大&#xff0c;从散点图的分布中越能看出来一些规律。 什么是散点图 在直角坐标系中&#xff0c;用两组数据构成多个坐标点&#xff0…