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

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  运行结果

 

猜你喜欢

转载自zhangweieye.iteye.com/blog/2348709
今日推荐