【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

相关文章

Spring Gateway动态路由实现方案

《SpringGateway动态路由实现方案》本文主要介绍了SpringGateway动态路由实现方案,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随... 目录前沿何为路由RouteDefinitionRouteLocator工作流程动态路由实现尾巴前沿S

JavaScript对象转数组的三种方法实现

《JavaScript对象转数组的三种方法实现》本文介绍了在JavaScript中将对象转换为数组的三种实用方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友... 目录方法1:使用Object.keys()和Array.map()方法2:使用Object.entr

idea+spring boot创建项目的搭建全过程

《idea+springboot创建项目的搭建全过程》SpringBoot是Spring社区发布的一个开源项目,旨在帮助开发者快速并且更简单的构建项目,:本文主要介绍idea+springb... 目录一.idea四种搭建方式1.Javaidea命名规范2JavaWebTomcat的安装一.明确tomcat

Java高效实现PowerPoint转PDF的示例详解

《Java高效实现PowerPoint转PDF的示例详解》在日常开发或办公场景中,经常需要将PowerPoint演示文稿(PPT/PPTX)转换为PDF,本文将介绍从基础转换到高级设置的多种用法,大家... 目录为什么要将 PowerPoint 转换为 PDF安装 Spire.Presentation fo

Java集合之Iterator迭代器实现代码解析

《Java集合之Iterator迭代器实现代码解析》迭代器Iterator是Java集合框架中的一个核心接口,位于java.util包下,它定义了一种标准的元素访问机制,为各种集合类型提供了一种统一的... 目录一、什么是Iterator二、Iterator的核心方法三、基本使用示例四、Iterator的工

SpringBoot中ResponseEntity的使用方法举例详解

《SpringBoot中ResponseEntity的使用方法举例详解》ResponseEntity是Spring的一个用于表示HTTP响应的全功能对象,它可以包含响应的状态码、头信息及响应体内容,下... 目录一、ResponseEntity概述基本特点:二、ResponseEntity的基本用法1. 创

springboot依靠security实现digest认证的实践

《springboot依靠security实现digest认证的实践》HTTP摘要认证通过加密参数(如nonce、response)验证身份,避免明文传输,但存在密码存储风险,相比基本认证更安全,却因... 目录概述参数Demopom.XML依赖Digest1Application.JavaMyPasswo

java中判断json key是否存在的几种方法

《java中判断jsonkey是否存在的几种方法》在使用Java处理JSON数据时,如何判断某一个key是否存在?本文就来介绍三种方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的... 目http://www.chinasem.cn录第一种方法是使用 jsONObject 的 has 方法

Java 线程池+分布式实现代码

《Java线程池+分布式实现代码》在Java开发中,池通过预先创建并管理一定数量的资源,避免频繁创建和销毁资源带来的性能开销,从而提高系统效率,:本文主要介绍Java线程池+分布式实现代码,需要... 目录1. 线程池1.1 自定义线程池实现1.1.1 线程池核心1.1.2 代码示例1.2 总结流程2. J

Java8 Collectors.toMap() 的两种用法

《Java8Collectors.toMap()的两种用法》Collectors.toMap():JDK8中提供,用于将Stream流转换为Map,本文给大家介绍Java8Collector... 目录一、简单介绍用法1:根据某一属性,对对象的实例或属性做映射用法2:根据某一属性,对对象集合进行去重二、Du