【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

相关文章

Java中ArrayList和LinkedList有什么区别举例详解

《Java中ArrayList和LinkedList有什么区别举例详解》:本文主要介绍Java中ArrayList和LinkedList区别的相关资料,包括数据结构特性、核心操作性能、内存与GC影... 目录一、底层数据结构二、核心操作性能对比三、内存与 GC 影响四、扩容机制五、线程安全与并发方案六、工程

JavaScript中的reduce方法执行过程、使用场景及进阶用法

《JavaScript中的reduce方法执行过程、使用场景及进阶用法》:本文主要介绍JavaScript中的reduce方法执行过程、使用场景及进阶用法的相关资料,reduce是JavaScri... 目录1. 什么是reduce2. reduce语法2.1 语法2.2 参数说明3. reduce执行过程

如何使用Java实现请求deepseek

《如何使用Java实现请求deepseek》这篇文章主要为大家详细介绍了如何使用Java实现请求deepseek功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1.deepseek的api创建2.Java实现请求deepseek2.1 pom文件2.2 json转化文件2.2

Java调用DeepSeek API的最佳实践及详细代码示例

《Java调用DeepSeekAPI的最佳实践及详细代码示例》:本文主要介绍如何使用Java调用DeepSeekAPI,包括获取API密钥、添加HTTP客户端依赖、创建HTTP请求、处理响应、... 目录1. 获取API密钥2. 添加HTTP客户端依赖3. 创建HTTP请求4. 处理响应5. 错误处理6.

Spring AI集成DeepSeek的详细步骤

《SpringAI集成DeepSeek的详细步骤》DeepSeek作为一款卓越的国产AI模型,越来越多的公司考虑在自己的应用中集成,对于Java应用来说,我们可以借助SpringAI集成DeepSe... 目录DeepSeek 介绍Spring AI 是什么?1、环境准备2、构建项目2.1、pom依赖2.2

Spring Cloud LoadBalancer 负载均衡详解

《SpringCloudLoadBalancer负载均衡详解》本文介绍了如何在SpringCloud中使用SpringCloudLoadBalancer实现客户端负载均衡,并详细讲解了轮询策略和... 目录1. 在 idea 上运行多个服务2. 问题引入3. 负载均衡4. Spring Cloud Load

Springboot中分析SQL性能的两种方式详解

《Springboot中分析SQL性能的两种方式详解》文章介绍了SQL性能分析的两种方式:MyBatis-Plus性能分析插件和p6spy框架,MyBatis-Plus插件配置简单,适用于开发和测试环... 目录SQL性能分析的两种方式:功能介绍实现方式:实现步骤:SQL性能分析的两种方式:功能介绍记录

在 Spring Boot 中使用 @Autowired和 @Bean注解的示例详解

《在SpringBoot中使用@Autowired和@Bean注解的示例详解》本文通过一个示例演示了如何在SpringBoot中使用@Autowired和@Bean注解进行依赖注入和Bean... 目录在 Spring Boot 中使用 @Autowired 和 @Bean 注解示例背景1. 定义 Stud

如何通过海康威视设备网络SDK进行Java二次开发摄像头车牌识别详解

《如何通过海康威视设备网络SDK进行Java二次开发摄像头车牌识别详解》:本文主要介绍如何通过海康威视设备网络SDK进行Java二次开发摄像头车牌识别的相关资料,描述了如何使用海康威视设备网络SD... 目录前言开发流程问题和解决方案dll库加载不到的问题老旧版本sdk不兼容的问题关键实现流程总结前言作为

SpringBoot中使用 ThreadLocal 进行多线程上下文管理及注意事项小结

《SpringBoot中使用ThreadLocal进行多线程上下文管理及注意事项小结》本文详细介绍了ThreadLocal的原理、使用场景和示例代码,并在SpringBoot中使用ThreadLo... 目录前言技术积累1.什么是 ThreadLocal2. ThreadLocal 的原理2.1 线程隔离2