行为设计模式 -命令模式- JAVA

命令模式

    • 一.简介
    • 二. 案例
      • 2.1 接收者(Receiver)
      • 2.2 命令接口实现对象(ConcreteCommand)
      • 2.3 调用者( invoker)
      • 2.4 获取Receiver对象
      • 2. 5 装配者客户端测试
    • 三. 结论
      • 3.1 要点
      • 3.2 示例

前言
本设计模式专栏写了很多设计模式的文章,希望大家点赞收藏一起学习
这是我在这个网站整理的笔记,有错误的地方请指出,关注我,接下来还会持续更新。

作者:神的孩子都在歌唱

一.简介

百度百科: 在软件系统中,行为请求者行为实现者通常呈现一种紧耦合。但在某些场合,比如要对行为进行记录、撤销/重做、事务等处理,这种无法抵御变化的紧耦合是不合适的。在这种情况下,如何将行为请求者与行为实现者解耦?将一组行为抽象为对象实现二者之间的松耦合。这就是命令模式(Command Pattern)。

个人理解: 某个行为有多个对象执行,并且我们需要对这个行为进行一些监控等处理。那么我们就需要将这种行为抽象出来,这种就叫做命令模式

在命令模式中有如下角色:

  • 接收者(Receiver): 任何类都可能成为一个接收者,只要它能够实现命令要求实现的相应功能。
  • 命令接口实现对象(ConcreteCommand): 是“虚”的实现;通常会持有接收者,并调用接收者的功能来完成命令要执行的操作。
  • 调用者(Invoker): 要求命令对象执行请求,通常会持有命令对象,可以持有很多的命令对象。这个是客户端真正触发命令并要求命令执行相应操作的地方,也就是说相当于使用命令对象的入口。
  • 装配者(Client): 创建具体的命令对象,并且设置命令对象的接收者。注意这个不是我们常规意义上的客户端,而是在组装命令对象和接收者。

接下来使用示例和代码来加深理解

二. 案例

举个例子:

  1. 场景:有两个人听从教练的指挥互相传球,一个用左手传,一个用右手传。
  2. 结论: 可以看出,我们有两个对象,简称为左手人和右手人,他们有一个共同的行为就是传球。由谁进行传球这个行为是教练下的命令。这时候就可以使用命令模式

类图如下

image-20240913104404498

2.1 接收者(Receiver)

我们首先定义两个人物对象

定义接收者的接口

/*** @Author chenyunzhi* @DATE 2024/9/12 15:13* @Description: 接收者*/
public interface BallReceiver {void passBall();
}

使用左手投球的人

/*** @Author chenyunzhi* @DATE 2024/9/12 15:14* @Description: 使用左手接收者*/
public class LeftHandPeopleReceiver implements BallReceiver{@Overridepublic void passBall() {System.out.println("人物1使用左手传球");}
}

使用右手投球的人

/*** @Author chenyunzhi* @DATE 2024/9/12 15:16* @Description: 使用右手接收者*/
public class RightHandPeopleReceiver implements BallReceiver{@Overridepublic void passBall() {System.out.println("人物2使用右手传球");}
}

2.2 命令接口实现对象(ConcreteCommand)

接下来我们要将传球这个命令抽象出来

定义执行命令的接口

/*** @Author chenyunzhi* @DATE 2024/9/12 15:19* @Description: 执行命令*/
public interface Command {void execute();
}

定义传球命令

/*** @Author chenyunzhi* @DATE 2024/9/12 15:23* @Description: 传球命令*/
public class PassBallCommand implements Command{private final BallReceiver ballReceiver;public PassBallCommand(BallReceiver ballReceiver) {this.ballReceiver = ballReceiver;}@Overridepublic void execute() {this.ballReceiver.passBall();}
}

现在我们已经准备好了接收者和命令实现,因此我们可以开始实现 调用者( invoker) 类。

2.3 调用者( invoker)

调用者的任务就是负责调用具体的命令

/*** @Author chenyunzhi* @DATE 2024/9/12 16:27* @Description: 命令的调度者,执行命令*/
public class BallInvoker {public Command command;public BallInvoker(Command command) {this.command = command;}public void execute() {this.command.execute();}
}

2.4 获取Receiver对象

命令模式已经准备就绪,我们可以开始编写一个简单的命令模式客户端程序。但在此之前,我将提供一个方法来创建相应的 BallReceiver 对象。我们可以通过工厂+策略模式去获取对应的人物对象

/*** @Author chenyunzhi* @DATE 2024/9/12 16:36* @Description: 使用工厂模式设计*/
public class PeopleReceiverFactory {public static BallReceiver getBallReceiver(String people) {switch (people) {case "LEFT_HAND":return new LeftHandPeopleReceiver();case "RIGHT_HAND":return new RightHandPeopleReceiver();default:return null;}}
}

如果命令很多,比如这里不止传球,还有打球,投球等等,这些行为可以通过工厂的方式获取。

2. 5 装配者客户端测试

准备工作已经完成,那么我们接下来就也可以开始测试了

/*** @Author chenyunzhi* @DATE 2024/9/12 16:45* @Description:*/
public class CommandPatternClient {/*** 装配者*/public static void ballClient(String receiver) {// 创建左手传球的接收者对象BallReceiver ballReceiver = PeopleReceiverFactory.getBallReceiver(receiver);// 创建命令并与接收者关联PassBallCommand passBallCommand = new PassBallCommand(ballReceiver);// 创建调用者与命令关联BallInvoker ballInvoker = new BallInvoker(passBallCommand);// 对调用者对象执行命令ballInvoker.execute();}public static void main(String[] args) {ballClient("LEFT_HAND");ballClient("RIGHT_HAND");}
}

ballClient方法负责组装命令对象和接收者。最后输出结果

image-20240913103844539

三. 结论

3.1 要点

  • 接收器实现与命令实现是分开的。
  • 命令实现类选择要在接收器对象上调用的方法,接收器中的每个方法都将有一个命令实现。它充当接收者和操作方法之间的桥梁。
  • Invoker 类只是将请求从 client 转发到 command 对象。
  • 客户端负责实例化适当的命令和接收器实现,然后将它们关联在一起。
  • Client 还负责实例化 invoker 对象并将 command object 与其关联,并执行 action 方法。
  • 命令设计模式很容易扩展,我们可以在接收器中添加新的动作方法,并在不改变客户端代码的情况下创建新的命令实现。
  • Command 设计模式的缺点是,由于有太多的关联,代码会变得庞大且令人困惑。

3.2 示例

java.lang.Runnable

javax.swing.Action

作者:神的孩子都在歌唱

本人博客:https://blog.csdn.net/weixin_46654114

转载说明:务必注明来源,附带本人博客连接。

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

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

相关文章

LCR 024. 反转链表 最细图片逐行解析过程

LCR 024. 反转链表 给定单链表的头节点 head ,请反转链表,并返回反转后的链表的头节点。 示例 1: 输入:head [1,2,3,4,5] 输出:[5,4,3,2,1]示例 2: 输入:head [1,2] 输出:[2,1]示例…

云计算平台上的DevOps实践

文章目录 什么是DevOps云计算平台上的DevOps优势自动化部署弹性伸缩地理分布 实施DevOps的关键组件版本控制系统持续集成/持续交付工具配置管理工具监控和日志管理 实践案例使用AWS CodePipeline进行持续集成/持续交付利用AWS Auto Scaling实现弹性使用AWS CloudFormation进行基…

DIY可视化-uniapp悬浮菜单支持拖动、吸附-代码生成器

在Uniapp中,悬浮菜单支持拖动和吸附功能,可以为用户带来更加灵活和便捷的操作体验。以下是对这两个功能的详细解释: 悬浮菜单支持拖动 提高用户体验:用户可以根据自己的需要,将悬浮菜单拖动到屏幕上的任意位置&#x…

微信小程序 - 动画(Animation)执行过程 / 实现过程 / 实现方式

前言 因官方文档描述不清晰,本文主要介绍微信小程序动画 实现过程 / 实现方式。 实现过程 推荐你对照 官方文档 来看本文章,这样更有利于理解。 简单来说,整个动画实现过程就三步: 创建一个动画实例 animation。调用实例的方法来描述动画。最后通过动画实例的 export 方法…

Kafka认证时Successfully logged in真的认证成功了?

背景 某个应用需要配置 Kafka 集群信息,且需要在验证集群是否可达。基本实现思路是创建一个生产者对象,然后发送一条测试数据,调用 Producer 的 send 方法发送消息后,再调用 get() 方法,即同步发送消息,测…

基于spootboot学生选课系统设计与实现

资料下载 https://download.csdn.net/download/qq_63753925/89888794 https://download.csdn.net/download/qq_63753925/89888793 https://download.csdn.net/download/qq_63753925/89885091 https://download.csdn.net/download/qq_63753925/89882320 摘 要 近年来&#xf…

机器人转人工时,开启实时质检(mod_cti基于FreeSWITCH)

文章目录 前言联系我们实现步骤1. 修改拨号方案2. 启用拨号方案 前言 在客户与机器人对话中,是不能开启质检功能的。因为机器人识别会与质检识别产生冲突。如果用户想通过机器人转接到人工时,开启质检功能,记录客户与人工之间的对话。应该如…

MySQL史上最全总结

MySQL学习笔记 安装与配置myini文件内容:初始化MySQL: MySQL语法:SQL-DDL数据库1.创建数据库2.查看数据库3.修改4.删除 数据库中的表管理1.创建表2.查找3.修改4.删除5.截断表 SQL-DML1.添加数据1.1插入多条数据1.2表内容复制 2.修改数据3.删除 TRUNCATE和…

Android中的epoll机制

深入理解Android中的epoll机制 在Android系统中,epoll广泛用于高效管理网络和文件的I/O操作。它通过减少CPU资源消耗和避免频繁的内核态-用户态切换,实现了在多连接、多任务环境中的高性能。epoll的特性使其非常适合Android系统中网络服务器、Socket通信…

php伪协议和move_uploaded_file、rename、copy等文件操作

move_uploaded_file、rename、copy 三个函数的区别: move_uploaded_file 函数是专门用于将通过 HTTP 上传的临时文件移动到指定位置的。如果你想要将一个已经存在的文件移动到另一个位置,而不是上传的文件,那么你应该使用 rename 函数或 co…

Java SPI 机制详解

面向对象设计鼓励模块间基于接口而非具体实现编程,以降低模块间的耦合,遵循依赖倒置原则,并支持开闭原则(对扩展开放,对修改封闭)。然而,直接依赖具体实现会导致在替换实现时需要修改代码&#…

自动驾驶性能分析时,非常有用的两个信息

自动驾驶的关键路径如下,传感器的数据发送给感知模块;感知模块根据传感器数据来确定车辆所处的环境,比如前方有没有障碍物,是不是和车道线保持着适当的距离等;感知处理之后的数据传递给规控模块,规控根据车…

【Spring】Cookie与Session

💐个人主页:初晴~ 📚相关专栏:计算机网络那些事 一、Cookie是什么? Cookie的存在主要是为了解决HTTP协议的无状态性问题,即协议本身无法记住用户之前的操作。 "⽆状态" 的含义指的是: 默认情况…

【大模型系列】Mini-InternVL(2024.10)

Paper:https://arxiv.org/pdf/2410.16261Github:https://github.com/OpenGVLab/InternVL/tree/main/internvl_chat/shell/mini_internvlAuthor:Zhangwei Gao et al. 上海人工智能实验室 文章目录 0 总结(省流版)1 模型结构1.1 InternViT-300M…

MySQL 数据库备份与恢复全攻略

MySQL 数据库备份与恢复全攻略 引言 在现代应用中,数据库是核心组件之一。无论是个人项目还是企业级应用,数据的安全性和完整性都至关重要。为了防止数据丢失、损坏或意外删除,定期备份数据库是必不可少的。本文将详细介绍 MySQL 数据库的备…

大语言模型数据流程源码解读(基于llama3模型)

文章目录 前言一、数据进入LlamaForCausalLM(LlamaPreTrainedModel)类二、数据进入LlamaModel(LlamaPreTrainedModel)类1、input_ids的embedding编码2、position_ids位置获取3、causal_mask因果mask构建1、causal_mask调用2、因果mask代码解读(_update_causal_mask)4、hidden_s…

Docker镜像的创建、修改与导出

Docker镜像的创建、修改与导出 前言一、创建Docker镜像1. 基于已有镜像创建方法一:修改现有镜像方法二:使用Dockerfile通过源码编译安装nginx二、修改Docker镜像1. 基于已有镜像创建新镜像方法一:修改现有镜像2. 使用`docker commit`命令创建新镜像方法一:提交正在运行的容…

Depcheck——专门用于检测 JavaScript 和 Node.js 项目中未使用依赖项的工具

文章目录 Depcheck 是什麽核心功能📚检测未使用的依赖🐛检测缺失的依赖✨支持多种文件类型🌍可扩展性 安装与使用1. 安装 Depcheck2. 使用 Depcheck Depcheck 的应用总结项目源码: Depcheck 是什麽 来看一个常见错误场景&#x1…

Sqoop的安装配置及使用

Sqoop安装前需要检查之前是否安装了Tez,否则会产生版本或依赖冲突,我们需要移除tez-site.xml,并将hadoop中的mapred-site.xml配置文件中的mapreduce驱动改回成yarn,然后分发到其他节点,hive里面配置的tez也要移除,然后…

sqoop抽取数据报驱动包不存在的问题

sqoop抽取数据报驱动包不存在的问题 报错示例:需要把相应的jar包放到sqoop的lib目录下: 可以正常查询