在程序设计中,经常设计到一个对象需要请求另外一个对象调用其方法达到某种目的,如果请求这不希望或不直接和被请求者打交道,既请求者不包含被请求者的引用,那么就可以使用命令模式。例如在军队中,指挥官请求三连偷袭敌人,但是指挥官不希望或无法直接与三连取得联系,那么可以将该请求形成一个命令,该命令的核心是让三连偷袭敌人。只要能让该命令被执行,就会实现偷袭敌人的目的。
命令模式是关于怎样处理一个对象请求到另一个对象调用其方法完成某项任务的一种成熟的模式,这里称提出请求的对象为请求者,被请求的对象为接收者。在命令模式中,当一个对象请求另一个对象其调用方法时,不和被请求的对象直接打交道,而是把这这种请求封装到一个命令的对象中,起封装的手段是将请求封装在命令对象的一个方法中。命令模式的核心就是使用命令对象来封装方法调用。既请求者的请求,接受者调用方法。
结构:
命令者模式包括四种角色
接收者:接受者是一个类的实例,该实例负责执行与请求相关的操作
命令:命令是一个接口,规定了用来封装请求的若干个方法。
具体命令:具体命令是实现命令接口的类的实例,具体命令必须实现命令接口中的方法。
请求者:请求者是一个包含”命令接口“变量的类的实例。请求者中的”命令“接口的变量可以存放任何具体命令的引用,请求者负责调用具体命令,让具体命令执行那些封装了请求的方法
代码实例:
接受者
- /**
- * 接受者:具体执行命令的人
- * @author hnylchf
- *
- */
- public class Receiver {
- public void sendMessage(){
- System.out.println("保证完成任务");
- }
- }
命令
- package org.zhy.command;
- /**
- * 命令,封装请求的方法
- * @author hnylchf
- *
- */
- public interface Command {
- public void exceute();
- }
请求者
- package org.zhy.command;
- /**
- * 请求者
- *
- * @author hnylchf
- *
- */
- public class Invoker {
- Command command;
- public void startExceuteCommand(Command command) {
- this.command = command;
- command.exceute();
- }
- }
具体命令
- package org.zhy.command;
- /**
- * 具体命令
- *
- * @author hnylchf
- *
- */
- public class ConcreteCommand {
- Receiver receiver;
- public ConcreteCommand(Receiver receiver) {
- this.receiver = receiver;
- }
- public void execute() {
- receiver.sendMessage();
- }
- }
演示:
- package org.zhy.command;
- public class Appliction {
- public static void main(String[] args) {
- Receiver receiver = new Receiver(); //执行者
- Command command = new ConcreteCommand(receiver); //命令和具体命令
- Invoker invoker = new Invoker(); //请求者
- invoker.startExceuteCommand(command); //请求者下达命令
- }
- }
命令接口的撤销方法
命令接口中规定了用来封装请求的方法比如上例的execute方法,命令接口中还可以提供用来封装撤销请求的方法。既撤销方法的执行能撤销execute方法的执行效果。如果execute方法的执行效果不可撤销,那么具体命令就不必实现撤销方法。
以下示例一个关于撤销方法的demo
命令:
- package org.zhy.commandundo;
- public interface Command {
- public void execute(String name);
- public void undo();
- }
具体命令:
- package org.zhy.commandundo;
- import java.util.ArrayList;
- public class ConcreteCommand implements Command {
- ArrayList<String> dirNameList;
- MakeDir dir;
- public ConcreteCommand(MakeDir dir) {
- dirNameList = new ArrayList<String>();
- this.dir = dir;
- }
- @Override
- public void execute(String name) {
- dir.createDir(name);
- dirNameList.add(name); //记录添加到list中
- }
- @Override
- public void undo() {
- if (dirNameList != null && dirNameList.size() > 0) {
- String str = dirNameList.get(dirNameList.size() - 1);
- dir.deleteDir(str);
- dirNameList.remove(dirNameList.size() - 1);
- } else {
- System.out.println("没有需要操作的记录");
- }
- }
- }
执行者
- package org.zhy.commandundo;
- import java.io.File;
- public class MakeDir {
- public void createDir(String name) {
- File file = new File(name);
- file.mkdir();
- }
- public void deleteDir(String name) {
- File file = new File(name);
- file.delete();
- }
- }
请求者
- package org.zhy.commandundo;
- public class RequestMakedir {
- Command command;
- public RequestMakedir(Command command) {
- this.command = command;
- }
- public void startExecuteCommand(String name) {
- command.execute(name);
- }
- public void undoExecuteCommand() {
- command.undo();
- }
- }
演示
- package org.zhy.commandundo;
- public class Appliction {
- public static void main(String[] args) {
- MakeDir dir = new MakeDir();
- Command command = new ConcreteCommand(dir);
- RequestMakedir requestMakedir = new RequestMakedir(command);
- requestMakedir.startExecuteCommand("jiafei");
- requestMakedir.startExecuteCommand("jiafei2");
- requestMakedir.startExecuteCommand("jiafei3");
- requestMakedir.undoExecuteCommand();
- }
- }
宏命令:
宏命令也是一个具体的命令,只不过他包含了其他命令的引用。当一个宏命令执行其execute方法时,将导致所引用的其他命令执行,因此执行一个宏命令相当于执行了多具体命令。
请求者输入英文字母或1-n之间的偶数可以请求既输出英文字母表,又输出1-n之间的偶数。
首先设计两个接收者,一个是输出英文字母表,一个是输出1-n之间的偶数
接收者1:打印英文字母
- package org.zhy.commandmacro;
- public class PrintLetter {
- public void printEnglish() {
- for (char c = 'a'; c < 'z'; c++) {
- System.out.print(" " + c);
- }
- }
- }
接受者2:打印偶数
- package org.zhy.commandmacro;
- public class PrintNumber {
- int n;
- public PrintNumber(int n) {
- this.n = n;
- }
- public void printEventNumber() {
- for (int i = 0; i <= n; i++) {
- if (i % 2 == 0) {
- System.out.print(" " + i);
- }
- }
- }
- }
命令:
- package org.zhy.commandmacro;
- public interface Command {
- public void execute();
- }
命令1:处理打印字母
- package org.zhy.commandmacro;
- public class PrintEnglishCommand implements Command {
- PrintLetter letter;
- public PrintEnglishCommand(PrintLetter letter) {
- this.letter = letter;
- }
- @Override
- public void execute() {
- letter.printEnglish();
- }
- }
命令2:处理打印偶数
- package org.zhy.commandmacro;
- public class PrintNumberCommand implements Command {
- PrintNumber number;
- public PrintNumberCommand(PrintNumber number) {
- this.number = number;
- }
- @Override
- public void execute() {
- number.printEventNumber();
- }
- }
命令3:宏命令处理一批命令(打印字母和打印偶数)
- package org.zhy.commandmacro;
- import java.util.ArrayList;
- public class MacroCommand implements Command {
- ArrayList<Command> commandList;
- public MacroCommand(ArrayList<Command> commandList) {
- this.commandList = commandList;
- }
- @Override
- public void execute() {
- for (Command command : commandList) {
- command.execute();
- }
- }
- }
请求者
- package org.zhy.commandmacro;
- public class RequestPerson {
- Command command;
- public Command getCommand() {
- return command;
- }
- public void setCommand(Command command) {
- this.command = command;
- }
- public void startExecuteCommand() {
- command.execute();
- }
- }
演示
- package org.zhy.commandmacro;
- import java.util.ArrayList;
- public class Appliction {
- public static void main(String[] args) {
- ArrayList<Command> commandList = new ArrayList<Command>();
- RequestPerson person = new RequestPerson();
- Command englishCommand = new PrintEnglishCommand(new PrintLetter());
- Command numberCommand = new PrintNumberCommand(new PrintNumber(10));
- commandList.add(englishCommand);
- commandList.add(numberCommand);
- Command macroCommand = new MacroCommand(commandList);
- person.setCommand(macroCommand);
- person.startExecuteCommand();
- }
- }
命令者模式的优点
1、在命令者模式中,请求者不直接与接受者互交,既请求者不包含接受者的引用,因此彻底消除了彼此之间的耦合。
2、命令者模式满足了软件的“开-闭原则”。如果增加新的具体命令和该命令的接受者,不必修改调用者的代码,调用者就可以直接使用新的命令对象。反之如果增加新的调用者,不必修改现有的具体命令和接受者。新增加的调用者就可以使用已有的具体命令
3、由于请求者的请求被封装到了具体命令中,那么就可以将具体命令保存到持久化媒介中,在需要的时候重新执行这个具体命令。因此使用命令者模式可以记录日志
4、使用命令者模式可以对请求者的请求进行排队,每个请求者各自对应一个具体命令,因此可以按一定的顺序执行这些命令。
适用命令者模式的场景
1、程序需要在不同的时刻指定,排列和执行请求
2、程序需要提供撤销操作
3、程序需要支持宏操作
相关推荐
根据给定文件内容,以下是关于Java设计模式的知识点说明: 1. 策略模式(Strategy Pattern)是一种行为设计模式,允许在运行时选择算法的行为。策略模式的意图是定义一系列算法,将每个算法封装起来,并使它们可以...
JAVA-设计模式-行为型模式-命令模式
这个“JAVA设计模式-chm版”资源显然包含了关于Java设计模式的详细信息,便于理解和应用。设计模式是对常见问题的解决方案的标准化描述,它们在软件工程中起到了重要的作用,帮助开发者创建可维护、可扩展且易于理解...
Java设计模式是软件工程中的一种重要思想,它总结了在解决特定问题时,程序员们反复使用的一些最佳实践和解决方案。这个资源"Java设计模式----通俗易懂版"显然是一个专门针对初学者或需要深入理解设计模式的开发者...
设计模式(Design pattern)代表了最佳的实践,通常被有经验的面向对象的软件开发人员所采用。设计模式是软件开发人员在软件开发过程中面临的一般问题的解决方案。这些解决方案是众多软件开发人员经过相当长的一段...
命令模式是一种行为设计模式,它将请求封装为一个对象,从而使你可用不同的请求对客户进行参数化,对请求排队或记录请求日志,以及支持可撤销的操作。在Java编程中,命令模式的应用广泛且实用。 命令模式的核心是将...
在 Java 中,命令模式是一种行为设计模式,它将请求封装为对象,允许对客户端进行参数化,包括队列、请求和操作。该模式还支持可撤销的操作,增强了管理和执行命令的灵活性。 二、命令设计模式的别名 Action(动作)...
### Java设计模式详解 #### 创建型模式 **1.1 工厂方法(Factory Method)** 工厂方法模式定义了一个创建对象的接口,但允许子类决定实例化哪一个类。这使得一个类的实例化可以被推迟到其子类。这种模式在以下...
### Java设计模式的应用 #### 一、引言 在当今快速发展的软件开发领域,Java作为一门功能强大且灵活的语言,不仅拥有丰富的API资源,还能与强大的数据库系统无缝对接。这使得许多开发人员能够以模块化的形式构建...
本资源主要介绍了JAVA设计模式的原则和23种设计模式的总结。设计模式是软件设计中的一种解决方案,能够使软件系统更加灵活、可维护和可扩展。本资源首先介绍了设计模式的六大原则,包括单一责任原则、开闭原则、里氏...
通过阅读《设计模式-Java语言中的应用》,读者不仅可以掌握设计模式的基本概念和原理,还能学习到如何在实际项目中选择合适的设计模式,以及如何优雅地在Java代码中实现这些模式。这将有助于提升开发者的编程技巧,...
《Java与模式》是阎宏博士的一本经典著作,它深入浅出地介绍了如何在Java编程中应用设计模式。这本书不仅讲解了设计模式的基本概念,还涵盖了23种经典的GOF设计模式,并结合Java语言特性进行了详细的解释和实例演示...
在软件工程中,设计模式是一种被广泛接受的解决特定设计问题的...因此,深入学习和理解Java设计模式对于任何Java开发者来说都是至关重要的。通过文档中的图解和代码示例,读者可以更好地理解和掌握这些模式的实际应用。
Java设计模式是软件工程中的一种重要思想,它代表了在解决特定问题时的最佳实践。这些模式都是经过时间和经验验证的设计解决方案,可以提高代码的可读性、可维护性和复用性。Java设计模式主要分为三类:创建型模式、...
**Java设计模式——入门** 设计模式是软件开发中的一种最佳实践,它是在特定上下文中解决常见问题的经验总结。Java设计模式是面向对象编程中的一个重要概念,可以帮助开发者编写可复用、可维护、易于理解的代码。这...
Java 命令模式设计模式详解 命令模式(Command Pattern)是一种行为设计模式,它允许将请求封装为一个对象,从而使不同的请求可以使用不同的参数进行参数化,队列或日志请求,以及支持可撤销的操作。在 Java 中,...
《Java与模式---闫宏》这本书是针对Java程序员深入理解设计模式的重要参考资料。设计模式是软件工程中的一个重要概念,它们代表了在特定上下文中解决常见问题的最佳实践。这本书结合了中国的道家思想,以一种独特的...
Java设计模式是软件开发中的一种最佳实践,它们是解决常见问题的经验总结,旨在提高代码的可重用性、可维护性和可扩展性。本篇Java设计模式(疯狂Java联盟版)详细介绍了二十三种设计模式,包括创建型、结构型和行为...
### Java设计模式——个人总结 #### 一、概述 本文档总结了常见的Java设计模式及其应用,涵盖了十种经典的设计模式,并提供了相应的代码示例。这些设计模式旨在解决软件开发过程中遇到的一系列常见问题,帮助...
Java设计模式——这个“言情版”的表述,可能是以一种轻松幽默的方式讲解这些模式,使得学习过程更加有趣。下面我们将深入探讨Java设计模式的一些核心知识点。 1. **单例模式**:在Java中,单例模式确保一个类只有...