【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

相关文章

JVM 的类初始化机制

前言 当你在 Java 程序中new对象时,有没有考虑过 JVM 是如何把静态的字节码(byte code)转化为运行时对象的呢,这个问题看似简单,但清楚的同学相信也不会太多,这篇文章首先介绍 JVM 类初始化的机制,然后给出几个易出错的实例来分析,帮助大家更好理解这个知识点。 JVM 将字节码转化为运行时对象分为三个阶段,分别是:loading 、Linking、initialization

Spring Security 基于表达式的权限控制

前言 spring security 3.0已经可以使用spring el表达式来控制授权,允许在表达式中使用复杂的布尔逻辑来控制访问的权限。 常见的表达式 Spring Security可用表达式对象的基类是SecurityExpressionRoot。 表达式描述hasRole([role])用户拥有制定的角色时返回true (Spring security默认会带有ROLE_前缀),去

浅析Spring Security认证过程

类图 为了方便理解Spring Security认证流程,特意画了如下的类图,包含相关的核心认证类 概述 核心验证器 AuthenticationManager 该对象提供了认证方法的入口,接收一个Authentiaton对象作为参数; public interface AuthenticationManager {Authentication authenticate(Authenti

Spring Security--Architecture Overview

1 核心组件 这一节主要介绍一些在Spring Security中常见且核心的Java类,它们之间的依赖,构建起了整个框架。想要理解整个架构,最起码得对这些类眼熟。 1.1 SecurityContextHolder SecurityContextHolder用于存储安全上下文(security context)的信息。当前操作的用户是谁,该用户是否已经被认证,他拥有哪些角色权限…这些都被保

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

Java架构师知识体认识

源码分析 常用设计模式 Proxy代理模式Factory工厂模式Singleton单例模式Delegate委派模式Strategy策略模式Prototype原型模式Template模板模式 Spring5 beans 接口实例化代理Bean操作 Context Ioc容器设计原理及高级特性Aop设计原理Factorybean与Beanfactory Transaction 声明式事物

Java进阶13讲__第12讲_1/2

多线程、线程池 1.  线程概念 1.1  什么是线程 1.2  线程的好处 2.   创建线程的三种方式 注意事项 2.1  继承Thread类 2.1.1 认识  2.1.2  编码实现  package cn.hdc.oop10.Thread;import org.slf4j.Logger;import org.slf4j.LoggerFactory

高效+灵活,万博智云全球发布AWS无代理跨云容灾方案!

摘要 近日,万博智云推出了基于AWS的无代理跨云容灾解决方案,并与拉丁美洲,中东,亚洲的合作伙伴面向全球开展了联合发布。这一方案以AWS应用环境为基础,将HyperBDR平台的高效、灵活和成本效益优势与无代理功能相结合,为全球企业带来实现了更便捷、经济的数据保护。 一、全球联合发布 9月2日,万博智云CEO Michael Wong在线上平台发布AWS无代理跨云容灾解决方案的阐述视频,介绍了

JAVA智听未来一站式有声阅读平台听书系统小程序源码

智听未来,一站式有声阅读平台听书系统 🌟&nbsp;开篇:遇见未来,从“智听”开始 在这个快节奏的时代,你是否渴望在忙碌的间隙,找到一片属于自己的宁静角落?是否梦想着能随时随地,沉浸在知识的海洋,或是故事的奇幻世界里?今天,就让我带你一起探索“智听未来”——这一站式有声阅读平台听书系统,它正悄悄改变着我们的阅读方式,让未来触手可及! 📚&nbsp;第一站:海量资源,应有尽有 走进“智听