【Java设计模式】命令模式:增强灵活的命令执行

2024-08-30 05:20

本文主要是介绍【Java设计模式】命令模式:增强灵活的命令执行,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 【Java设计模式】命令模式:增强灵活的命令执行
    • 一、概述
    • 二、命令设计模式的别名
    • 三、命令设计模式的意图
    • 四、命令模式的详细解释及实际示例
    • 五、Java中命令模式的编程示例
    • 六、何时在Java中使用命令模式
    • 七、命令模式在Java中的实际应用
    • 八、命令模式的优点和权衡
    • 九、源码下载

【Java设计模式】命令模式:增强灵活的命令执行

一、概述

在Java中,命令模式是一种行为设计模式,它将请求封装为对象,允许对客户端进行参数化,包括队列、请求和操作。该模式还支持可撤销的操作,增强了管理和执行命令的灵活性。

二、命令设计模式的别名

  • Action(动作)
  • Transaction(事务)

三、命令设计模式的意图

命令设计模式是Java编程中使用的一种行为模式。它将请求封装为对象,允许对客户端进行参数化,包括队列、请求和操作。此模式还支持可撤销的操作,增强了命令执行的灵活性。

四、命令模式的详细解释及实际示例

  1. 实际示例
    • 想象一个智能家居系统,您可以通过中央应用程序控制灯光、恒温器和安全摄像头等设备。每个操作这些设备的命令都被封装为一个对象,使系统能够排队、顺序执行和在必要时撤销命令。这种方法将控制逻辑与设备实现解耦,允许在不改变核心应用程序的情况下轻松添加新设备或功能。这种灵活性和功能说明了命令设计模式在Java编程中的实际应用。
  2. 通俗解释
    • 将请求存储为命令对象允许在以后执行或撤销该操作。
  3. 维基百科解释
    • 在面向对象编程中,命令模式是一种行为设计模式,其中一个对象用于封装在以后的时间执行操作或触发事件所需的所有信息。

五、Java中命令模式的编程示例

在命令模式中,对象用于封装在以后的时间执行操作或触发事件所需的所有信息。此模式对于在应用程序中实现撤销功能特别有用。
在我们的示例中,一个“巫师”对一个“哥布林”施放法术。每个法术都是一个命令对象,可以执行和撤销,展示了Java中命令模式的核心原则。法术一个接一个地在哥布林身上执行。第一个法术使哥布林缩小,第二个法术使他隐形。然后巫师一个接一个地逆转法术。这里的每个法术都是一个可以撤销的命令对象。
让我们从“巫师”类开始。

@Slf4j
public class Wizard {private final Deque<Runnable> undoStack = new LinkedList<>();private final Deque<Runnable> redoStack = new LinkedList<>();public Wizard() {}public void castSpell(Runnable runnable) {runnable.run();undoStack.offerLast(runnable);}public void undoLastSpell() {if (!undoStack.isEmpty()) {var previousSpell = undoStack.pollLast();redoStack.offerLast(previousSpell);previousSpell.run();}}public void redoLastSpell() {if (!redoStack.isEmpty()) {var previousSpell = redoStack.pollLast();undoStack.offerLast(previousSpell);previousSpell.run();}}@Overridepublic String toString() {return "Wizard";}
}

接下来,我们有“哥布林”,他是法术的“目标”。

@Slf4j
@Getter
@Setter
public abstract class Target {private Size size;private Visibility visibility;public void printStatus() {LOGGER.info("{}, [size={}] [visibility={}]", this, getSize(), getVisibility());}public void changeSize() {var oldSize = getSize() == Size.NORMAL? Size.SMALL : Size.NORMAL;setSize(oldSize);}public void changeVisibility() {var visible = getVisibility() == Visibility.INVISIBLE? Visibility.VISIBLE : Visibility.INVISIBLE;setVisibility(visible);}
}
public class Goblin extends Target {public Goblin() {setSize(Size.NORMAL);setVisibility(Visibility.VISIBLE);}@Overridepublic String toString() {return "Goblin";}
}

最后,我们可以展示“巫师”施放法术的完整示例。

public static void main(String[] args) {var wizard = new Wizard();var goblin = new Goblin();goblin.printStatus();wizard.castSpell(goblin::changeSize);goblin.printStatus();wizard.castSpell(goblin::changeVisibility);goblin.printStatus();wizard.undoLastSpell();goblin.printStatus();wizard.undoLastSpell();goblin.printStatus();wizard.redoLastSpell();goblin.printStatus();wizard.redoLastSpell();goblin.printStatus();
}

以下是程序输出:

20:13:38.406 [main] INFO com.iluwatar.command.Target -- Goblin, [size=normal] [visibility=visible]
20:13:38.409 [main] INFO com.iluwatar.command.Target -- Goblin, [size=small] [visibility=visible]
20:13:38.409 [main] INFO com.iluwatar.command.Target -- Goblin, [size=small] [visibility=invisible]
20:13:38.409 [main] INFO com.iluwatar.command.Target -- Goblin, [size=small] [visibility=visible]
20:13:38.409 [main] INFO com.iluwatar.command.Target -- Goblin, [size=normal] [visibility=visible]
20:13:38.409 [main] INFO com.iluwatar.command.Target -- Goblin, [size=small] [visibility=visible]
20:13:38.409 [main] INFO com.iluwatar.command.Target -- Goblin, [size=small] [visibility=invisible]

六、何时在Java中使用命令模式

当您需要用操作参数化对象、支持撤销操作、或围绕基于原始操作构建的高级操作构建系统时,命令设计模式是适用的。它通常用于GUI按钮、数据库事务和宏记录。
当您想要:

  1. 用要执行的操作参数化对象,为过程语言中发现的回调提供面向对象的替代方案。命令可以注册并稍后执行。
  2. 在不同时间指定、排队和执行请求,允许命令独立于原始请求存在,甚至可以跨进程传输。
  3. 支持撤销功能,其中命令的执行操作存储状态并包括一个撤销操作来逆转先前的操作。这通过维护历史列表允许无限的撤销和重做功能。
  4. 记录更改以在系统崩溃后重新应用它们。通过向命令接口添加加载和存储操作,您可以维护更改的持久日志,并通过重新加载和重新执行此日志中的命令来恢复。
  5. 围绕基于原始操作构建的高级操作构建系统,这在基于事务的系统中很常见。命令模式通过提供调用和扩展操作的公共接口来模拟事务。
  6. 保留请求的历史记录。
  7. 实现回调功能。
  8. 实现撤销功能。

七、命令模式在Java中的实际应用

  1. 桌面应用程序中的GUI按钮和菜单项。
  2. 支持回滚的数据库系统和事务系统中的操作。
  3. 文本编辑器和电子表格等应用程序中的宏记录。
  4. java.lang.Runnable
  5. org.junit.runners.model.Statement
  6. Netflix Hystrix
  7. javax.swing.Action

八、命令模式的优点和权衡

优点:

  1. 将调用操作的对象与知道如何执行操作的对象解耦。
  2. 很容易添加新的命令,因为您不必更改现有的类。
  3. 您可以将一组命令组合成一个复合命令。

权衡:

  1. 为每个单独的命令增加了类的数量。
  2. 通过在发送者和接收者之间添加多个层,可能会使设计复杂化。

九、源码下载

命令模式示例代码下载

通过本文的介绍,相信大家对Java中的命令模式有了更深入的了解。在实际开发中,合理运用命令模式可以提高代码的灵活性和可扩展性,但需要注意其增加的类数量和可能带来的设计复杂性。

这篇关于【Java设计模式】命令模式:增强灵活的命令执行的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1119905

相关文章

一文详解如何在idea中快速搭建一个Spring Boot项目

《一文详解如何在idea中快速搭建一个SpringBoot项目》IntelliJIDEA作为Java开发者的‌首选IDE‌,深度集成SpringBoot支持,可一键生成项目骨架、智能配置依赖,这篇文... 目录前言1、创建项目名称2、勾选需要的依赖3、在setting中检查maven4、编写数据源5、开启热

Java对异常的认识与异常的处理小结

《Java对异常的认识与异常的处理小结》Java程序在运行时可能出现的错误或非正常情况称为异常,下面给大家介绍Java对异常的认识与异常的处理,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参... 目录一、认识异常与异常类型。二、异常的处理三、总结 一、认识异常与异常类型。(1)简单定义-什么是

Redis Cluster模式配置

《RedisCluster模式配置》:本文主要介绍RedisCluster模式配置,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录分片 一、分片的本质与核心价值二、分片实现方案对比 ‌三、分片算法详解1. ‌范围分片(顺序分片)‌2. ‌哈希分片3. ‌虚

SpringBoot项目配置logback-spring.xml屏蔽特定路径的日志

《SpringBoot项目配置logback-spring.xml屏蔽特定路径的日志》在SpringBoot项目中,使用logback-spring.xml配置屏蔽特定路径的日志有两种常用方式,文中的... 目录方案一:基础配置(直接关闭目标路径日志)方案二:结合 Spring Profile 按环境屏蔽关

基于Python实现一个Windows Tree命令工具

《基于Python实现一个WindowsTree命令工具》今天想要在Windows平台的CMD命令终端窗口中使用像Linux下的tree命令,打印一下目录结构层级树,然而还真有tree命令,但是发现... 目录引言实现代码使用说明可用选项示例用法功能特点添加到环境变量方法一:创建批处理文件并添加到PATH1

Java使用HttpClient实现图片下载与本地保存功能

《Java使用HttpClient实现图片下载与本地保存功能》在当今数字化时代,网络资源的获取与处理已成为软件开发中的常见需求,其中,图片作为网络上最常见的资源之一,其下载与保存功能在许多应用场景中都... 目录引言一、Apache HttpClient简介二、技术栈与环境准备三、实现图片下载与保存功能1.

SpringBoot排查和解决JSON解析错误(400 Bad Request)的方法

《SpringBoot排查和解决JSON解析错误(400BadRequest)的方法》在开发SpringBootRESTfulAPI时,客户端与服务端的数据交互通常使用JSON格式,然而,JSON... 目录问题背景1. 问题描述2. 错误分析解决方案1. 手动重新输入jsON2. 使用工具清理JSON3.

java中long的一些常见用法

《java中long的一些常见用法》在Java中,long是一种基本数据类型,用于表示长整型数值,接下来通过本文给大家介绍java中long的一些常见用法,感兴趣的朋友一起看看吧... 在Java中,long是一种基本数据类型,用于表示长整型数值。它的取值范围比int更大,从-922337203685477

java Long 与long之间的转换流程

《javaLong与long之间的转换流程》Long类提供了一些方法,用于在long和其他数据类型(如String)之间进行转换,本文将详细介绍如何在Java中实现Long和long之间的转换,感... 目录概述流程步骤1:将long转换为Long对象步骤2:将Longhttp://www.cppcns.c

SpringBoot集成LiteFlow实现轻量级工作流引擎的详细过程

《SpringBoot集成LiteFlow实现轻量级工作流引擎的详细过程》LiteFlow是一款专注于逻辑驱动流程编排的轻量级框架,它以组件化方式快速构建和执行业务流程,有效解耦复杂业务逻辑,下面给大... 目录一、基础概念1.1 组件(Component)1.2 规则(Rule)1.3 上下文(Conte