您的位置:首页 > 其它

设计模式学习笔记(一:命令模式)

2016-10-17 23:29 225 查看

1.1概述

在许多设计中,经常涉及到一个对象请求另一个对象调用其方法达到某种目的。如果请求者不希望或无法直接和被请求者打交道,即不希望或无法含有被请求者的引用,那么就可以使用命令模式。

例如,在军队作战中,指挥官请求三连偷袭敌人,但是指挥官不希望或无法直接与三连取得联系,那么可以将该请求:“三连偷袭敌人”形成一个“作战命令”,该作战命名的核心就是“三连偷袭敌人”。只要能让该“作战命令”被执行,就会实现偷袭敌人的目的。

命令模式的核心就是使用命令对象来封装方法调用,即将请求者的请求:“接收者调用方法”封装到命令对象的一个方法中,这样一来,当一个对象请求另一个对象调用方法来完成某项任务时,只需和命令对象打交道,即让命令对象调用封装了“请求”的那个方法即可。具体如下图1所示:



图1 用命令对象封装方法调用

1.2模式的结构

命令模式的结构中包括四种角色:

(1)接收者(Receiver):接收者是一个类的实例,该实例负责执行与请求者相关的操作。

(2)命令接口(Command):命令是一个接口,规定了用来封装“请求”的若干个方法,比如,execute()、undo()等方法。

(3)具体命令(ConcreteCommand):具体命令是实现命令接口的类的实例。具体命令必须实现命令接口中的方法,比如execute()方法,是该方法封装一个“请求”。

(4)请求者(Invoker):请求者是一个包含Command接口变量的类的实例。请求者中的Command接口的变量可以存放任何具体命令的引用。请求者负责调用具体命令,让具体命令执行那些封装了“请求”的方法,比如execute()方法。

命令模式结构的类图具体如下图2所示:



图2 命令模式结构的类图

1.3命令模式的优点

(1)在命令模式中,请求者(Invoker)不直接与接收者(Receiver)交互,即请求者(Invoker)不包含接收者(Receiver)的引用,因此彻底消除了彼此之间的耦合。

(2)命令模式满足“开-闭原则”。如果增加新的具体命令和该命令的接收者,不必修改调用者的代码,调用者就可以直接使用新的命令对象;反之,如果增加新的调用者,不必修改现有的具体命令和接收者,新增加的调用者就可以使用已有的具体命令。

(3)由于请求者的请求被封装到了具体命令中,那么就可以将具体命令保存到持久化的媒介中,在需要的时候,重新执行这个具体命令。因此,命令模式可以记录日志。

(4)使用命令模式可以对请求者的“请求”进行排队。每个请求都各自对应一个具体命令,因此可以按照一定顺序执行这些具体命令。

1.4适合使用命令模式的情景

(1)程序需要在不同的时刻指定、排列和执行请求。

(2)程序需要提供撤销操作。

(3)程序需要支持宏操作(宏命令是一个具体命令,不过它包含了其他具体命令的引用。即执行一个宏命令,相当于执行了许多具体命令)

1.5命令模式的使用

下面通过一个简单的实例,实现1.1概述中简单例子:在军队作战中,指挥官请求三连偷袭敌人。具体如下:

首先看一下本实例构建框架具体类和1.2模式的结构中类图的对应关系,如下图3所示:



图3 具体应用类图对应关系

(1)接收者(Receiver)对应的CompanyArmy类代码如下:

package com.liuzhen.one_command;

public class CompanyArmy {
//接受者执行如何偷袭敌人
public void sneakAttack(){
System.out.println("我们知道如何偷袭敌人,保证完成任务!!!");
}
}


(2)命令接口(Command)对应的Command接口代码如下:

package com.liuzhen.one_command;

public interface Command {

public abstract void execute();
}


(3)具体命令(ConcreteCommand)对应的ConcreteCommand类对应的代码如下:

package com.liuzhen.one_command;

public class ConcreteCommand implements Command {

CompanyArmy army;   //创建接受者引用

ConcreteCommand(CompanyArmy army){
this.army = army;
}
//封装指挥官的请求
public void execute(){
army.sneakAttack();       //偷袭敌人
}
}


(4)请求者(Invoker)对应的ArmySuperior类代码如下:

package com.liuzhen.one_command;

public class ArmySuperior {

Command command;        //创建命令的引用
//用来存放具体命令的引用
public void setCommand(Command command){
this.command = command;
}
//让具体命令执行execute()方法,偷袭敌人
public void startExecuteCommand(){
command.execute();
}
}


(5)具体调用命令实现,指挥官请求三连偷袭敌人的类OneApplication类代码如下:

package com.liuzhen.one_command;

public class OneApplication {

/**
* @param args
*/
public static void main(String[] args) {
CompanyArmy 三连 = new CompanyArmy();     //创建接受者:三连
Command command = new ConcreteCommand(三连);  //创建具体命令,并指定接受者为三连
ArmySuperior 指挥官 = new ArmySuperior();  //创建请求者:指挥官
指挥官.setCommand(command);   //指挥者请求命令:三连偷袭敌人
指挥官.startExecuteCommand();  //指挥官确定执行命令:三连偷袭敌人

}

}


运行结果如下图4:





图4 运行结果

[b]参考资料:[/b]

1.Java设计模式/耿祥义,张跃平著.——北京:清华大学出版社,2009.5
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: