设计模式篇---命令模式(结合spring+动态代理实现开闭)

本文主要是介绍设计模式篇---命令模式(结合spring+动态代理实现开闭),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

    • 概念
    • 结构
    • 实例
    • 总结

概念

命令模式:将一个请求封装为一个对象,从而可用不同的请求对客户进行参数化,对请求排队或者记录请求日志,以及支持可撤销的操作。
现实生活中,我们用开关来控制一些电器的打开和关闭,比如电灯和电视。购买开关时,我们也不知道它未来用来控制哪些电器,也就是说开关和电灯、电视之间没有直接关系。安装开关可能用来控制电灯,也可能用来控制其他电器,取决于控制哪些电器的是电线,相同的开关可以通过不同的电线来控制不同的电器。
上面的例子中,开关可以理解为调用者,电线为命令类,电器为接收者。开关和电器之间并不存在耦合,想控制哪个电器,更换一根电线就可以了。

结构

在这里插入图片描述
Command(抽象命令类):其中声明了用于执行请求的execute()等方法,这些方法可以调用请求接收者的相关操作。
ConcreteCommand(具体命令类):抽象命令类的子类,它对应具体的接收者对象,将具体接收者对象绑定其中,当实现execute()方法时,将调用接收者对象的相关操作。
Invoker(调用者):它用来发送请求,关联抽象命令对象。
Receiver(接收者):接收者执行与请求相关的操作,具体实现对请求的业务处理。

实例

某系统实现一个功能,用一个按钮控制不同的功能。可以通过按钮退出系统,也可以通过按钮来显示帮助文档。
在这里插入图片描述

抽象命令类

public interface  Command {public  void execute();
}

请求发送者

public class FunctionButton {private Command command;public void setCommand(Command command) {this.command = command;}public void click() {System.out.println("点击功能键:");command.execute();}
}

退出命令类,充当具体命令类

public class ExitCommand implements Command {private SystemExitClass seObj;public ExitCommand() {seObj = new SystemExitClass();}@Overridepublic void execute() {seObj.exit();}
}

帮助命令类,充当具体命令类

public class HelpCommand implements Command {private DisplayHelpClass hcObj;public HelpCommand() {hcObj = new DisplayHelpClass();}@Overridepublic void execute() {hcObj.display();}
}

退出系统模拟实现类,充当请求接收者

public class SystemExitClass {public void exit() {System.out.println("退出系统");}
}

显示帮助文档模拟实现类,充当请求接受者

public class DisplayHelpClass {public void display() {System.out.println("显示帮助文档");}
}

客户端

public class Client {public static void main(String[] args) {FunctionButton button = new FunctionButton();button.setCommand(new ExitCommand());button.click();System.out.println("------------------");button.setCommand(new HelpCommand());button.click();}
}

运行结果
在这里插入图片描述

当我们在项目开发中,更多的是结合Spring使用,当客户端调用时,具体命令类发生改变时,我们尽量不去改动客户端代码,实现的方式有很多,上一篇刚写完代理模式,这里可以结合动态代理来实现bean的切换。

简单写一个config类,不足的地方的大家可自行扩展。

@Configuration
public class BeanConfig {@Resourceprivate ExitCommand exitCommand;@Resourceprivate HelpCommand helpCommand;private SwitcherInvocationHandler switcherInvocationHandler = new SwitcherInvocationHandler();class SwitcherInvocationHandler implements InvocationHandler {private final Set<String> methodsOnObjectClass = Arrays.stream(Object.class.getMethods()).map(Method::getName).collect(Collectors.toSet());@Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {if (methodsOnObjectClass.contains(method.getName())) {return method.invoke(this, args);}if (useExitCommand()) {return method.invoke(exitCommand, args);} else {return method.invoke(helpCommand, args);}}//需要变动的地方是这里private boolean useExitCommand() {return true;}}//通过反射创建名字为"command"的bean@Bean(name = "command")public Command getCommand() {return Reflection.newProxy(Command.class, switcherInvocationHandler);}
}

然后给各个类加上spring的注解

FunctionButton类,去掉set注入,改用Resource注解,拿取名为“command” 的bean。

@Service
public class FunctionButton {@Resourceprivate Command command;public void click() {System.out.println("点击功能键:");command.execute();}
}

ExitCommand 类

@Service
public class ExitCommand implements Command {private SystemExitClass seObj;public ExitCommand() {seObj = new SystemExitClass();}@Overridepublic void execute() {seObj.exit();}
}

HelpCommand类

@Service
public class HelpCommand implements Command {private DisplayHelpClass hcObj;public HelpCommand() {hcObj = new DisplayHelpClass();}@Overridepublic void execute() {hcObj.display();}
}

客户端直接注入发送者即可

@Service("commandClient")
public class CommandClient {@Autowiredprivate FunctionButton button;public void invoke() {button.click();}
}

写个单元测试调用一下

public class TestDemo {@Autowiredprivate CommandClient commandClient;@Testpublic void test() throws InterruptedException {commandClient.invoke();}}

调用结果:
在这里插入图片描述

总结

命令模式可以降低系统的耦合度,让请求者和接收者完全解耦,并且如果有新的命令加进来,也不用修改之前的代码,符合开闭原则。
命令模式和外观模式有些类似,都是通过中间一个对象进行解耦。命令模式更适合操作的切换,比如开关用来开灯,也可以用来开电视,开关作为调用者,可以调用开灯的命令,也可以调用开电视的命令,让调用者和接收者解耦。
外观模式更类似一组操作的集合,比如到家后,先换拖鞋,再开电视,最后坐在沙发上,这一系列的操作,可以放在外观层实现,让调用者和具体的子系统解耦。

这篇关于设计模式篇---命令模式(结合spring+动态代理实现开闭)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Java中的String.valueOf()和toString()方法区别小结

《Java中的String.valueOf()和toString()方法区别小结》字符串操作是开发者日常编程任务中不可或缺的一部分,转换为字符串是一种常见需求,其中最常见的就是String.value... 目录String.valueOf()方法方法定义方法实现使用示例使用场景toString()方法方法

Java中List的contains()方法的使用小结

《Java中List的contains()方法的使用小结》List的contains()方法用于检查列表中是否包含指定的元素,借助equals()方法进行判断,下面就来介绍Java中List的c... 目录详细展开1. 方法签名2. 工作原理3. 使用示例4. 注意事项总结结论:List 的 contain

MySQL双主搭建+keepalived高可用的实现

《MySQL双主搭建+keepalived高可用的实现》本文主要介绍了MySQL双主搭建+keepalived高可用的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,... 目录一、测试环境准备二、主从搭建1.创建复制用户2.创建复制关系3.开启复制,确认复制是否成功4.同

Java实现文件图片的预览和下载功能

《Java实现文件图片的预览和下载功能》这篇文章主要为大家详细介绍了如何使用Java实现文件图片的预览和下载功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... Java实现文件(图片)的预览和下载 @ApiOperation("访问文件") @GetMapping("

C#如何动态创建Label,及动态label事件

《C#如何动态创建Label,及动态label事件》:本文主要介绍C#如何动态创建Label,及动态label事件,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录C#如何动态创建Label,及动态label事件第一点:switch中的生成我们的label事件接着,

Spring Boot + MyBatis Plus 高效开发实战从入门到进阶优化(推荐)

《SpringBoot+MyBatisPlus高效开发实战从入门到进阶优化(推荐)》本文将详细介绍SpringBoot+MyBatisPlus的完整开发流程,并深入剖析分页查询、批量操作、动... 目录Spring Boot + MyBATis Plus 高效开发实战:从入门到进阶优化1. MyBatis

SpringCloud动态配置注解@RefreshScope与@Component的深度解析

《SpringCloud动态配置注解@RefreshScope与@Component的深度解析》在现代微服务架构中,动态配置管理是一个关键需求,本文将为大家介绍SpringCloud中相关的注解@Re... 目录引言1. @RefreshScope 的作用与原理1.1 什么是 @RefreshScope1.

Java并发编程必备之Synchronized关键字深入解析

《Java并发编程必备之Synchronized关键字深入解析》本文我们深入探索了Java中的Synchronized关键字,包括其互斥性和可重入性的特性,文章详细介绍了Synchronized的三种... 目录一、前言二、Synchronized关键字2.1 Synchronized的特性1. 互斥2.

MyBatis 动态 SQL 优化之标签的实战与技巧(常见用法)

《MyBatis动态SQL优化之标签的实战与技巧(常见用法)》本文通过详细的示例和实际应用场景,介绍了如何有效利用这些标签来优化MyBatis配置,提升开发效率,确保SQL的高效执行和安全性,感... 目录动态SQL详解一、动态SQL的核心概念1.1 什么是动态SQL?1.2 动态SQL的优点1.3 动态S

Spring Boot 配置文件之类型、加载顺序与最佳实践记录

《SpringBoot配置文件之类型、加载顺序与最佳实践记录》SpringBoot的配置文件是灵活且强大的工具,通过合理的配置管理,可以让应用开发和部署更加高效,无论是简单的属性配置,还是复杂... 目录Spring Boot 配置文件详解一、Spring Boot 配置文件类型1.1 applicatio