命令(Command)模式:又称Action模式或者Transaction模式。它属于对象的行为模式。命令模式把一个请求或者操作封装到一个对象中。命令模式允许系统使用不同的请求把客户端参数化,对请求排队或者记录请求日志,可以提供命令的撤销和撤销功能。
GoF命令模式结构图如下:
命令模式是有以下角色:
抽象命令(Command)角色:声明执行操作的接口。
具体命令(ConcreteCommand)角色:将一个接收者对象绑定到一个动作上。调用接收者相应的操作,以实现Execute方法。
客户端(Client)角色:创建一个命令对象并设定它的接收者。
请求者(Invoker)角色:负责调用命令对象的执行请求;
接收者(Receiver)角色:负责具体实施和执行一个请求相关的操作。任何一个类都可以作为一个接收者。
上面模型的模拟代码如下:
- package command;
- /**
- *
- *作者:alaric
- *时间:2013-8-20下午7:39:48
- *描述:接收者角色
- */
- public class Receiver {
-
- public Receiver() {
- super();
- }
-
- /**
- *
- *作者:alaric
- *时间:2013-8-20下午7:40:00
- *描述:行动方法
- */
- public void action(){
- System.out.println("接收者接到命令,开始行动");
- }
- }
- package command;
- /**
- *
- *作者:alaric
- *时间:2013-8-20下午7:36:51
- *描述:抽象命令角色
- */
- public interface Command {
-
- /**
- *
- *作者:alaric
- *时间:2013-8-20下午7:36:40
- *描述:执行方法
- */
- public void execute();
- }
- package command;
- /**
- *
- *作者:alaric
- *时间:2013-8-20下午7:37:10
- *描述:具体命令角色
- */
- public class ConcreteCommand implements Command {
-
- private Receiver receiver;
-
-
- public ConcreteCommand(Receiver receiver) {
- super();
- this.receiver = receiver;
- }
-
-
- @Override
- public void execute() {
-
- receiver.action();
-
- }
-
-
- }
- package command;
- /**
- *
- *作者:alaric
- *时间:2013-8-20下午7:41:07
- *描述:
- */
- public class Invoker {
-
- private Command command;
-
- public Invoker(Command command) {
- super();
- this.command = command;
- }
-
- public void action(){
- command.execute();
- }
- }
- package command;
- /**
- *
- *作者:alaric
- *时间:2013-8-20下午7:33:51
- *描述:客户端角色
- */
- public class Client {
-
- /**
- *作者:alaric
- *时间:2013-8-20下午7:33:44
- *描述:
- */
- public static void main(String[] args) {
- Receiver receiver = new Receiver();
- Command command = new ConcreteCommand(receiver);
- Invoker invoker = new Invoker(command);
- invoker.action();
- }
-
- }
命令模式是对命令的封装,它把发出命令的责任和执行命令的责任分隔开,委派给不同的对象。每一个命令都是一个操作,请求的一方发出请求要求执行一个操作;接收的一方收到请求并执行操作。命令模式允许请求的一方和接收的一方独立开来,使得请求的一方不必知道接收请求的一方接口,更不必知道请求是怎么被接收,以及操作是否被执行,何时被执行,以及是怎么被执行的。命令允许请求方和接收方各自都能独立演化,从而具有以下优点:
1、命令模式使新的命令在不改变现有结构代码的情况下很容易被加入到系统里;
2、允许接收请求的一方决定是否否决请求;
3、能较容易地设计一个命令队列;
4、可以容易地实现对请求的Undo和Redo操作;
5、在需要的情况下以较容易地将命令记入日志。
在讲门面模式的时候曾经提过一个例子,我们小时玩过投币的那种游戏机,一个方向操作杆,四个动作按钮,在一个操作面板上,封装了复杂的逻辑在机器内部,提供简单的操作界面,是一个门面的例子,然而每个操作发出一个命令,来控制游戏人物的运动和各种动作。方向操作杆是一个移动的命令,传入移动的方向和距离作为参数,还有出拳按键发出出拳命令,脚踢按键发出脚踢的命令,那还有组合操作命令,比如下踢腿(操作杆向下和踢脚按钮按下)。现在我们用命令模式来模拟这个场景。类的模型图如下:
如上图所示:Client相当于小时候的我们,Control相关于控制执行器,我们可以控制踢(KickCommand)、打(FistCommand)、移动(MoveCommand),这些命令的最终实现者也是接收者是游戏机(GameMachine)。
代码如下:
- package command.example;
- /**
- *
- *作者:alaric
- *时间:2013-8-21上午7:15:53
- *描述:接收者
- */
- public class GameMachine {
-
- public void fist() {
- System.out.println("出拳");
- }
-
- public void kick() {
- System.out.println("出脚");
- }
-
- public void move(String direction){
- System.out.println("向"+direction+"移动");
- }
-
- }
- package command.example;
- /**
- *
- *作者:alaric
- *时间:2013-8-20下午10:20:35
- *描述:命令接口
- */
- public interface Command {
-
- //执行方法
- public void execute();
-
- //这里还可以加入撤销方法,回滚方法等
- }
- package command.example;
- /**
- *
- *作者:alaric
- *时间:2013-8-21上午7:17:02
- *描述:拳打
- */
- public class FistCommand implements Command {
- private GameMachine machine;
-
-
-
- public FistCommand(GameMachine machine) {
- super();
- this.machine = machine;
- }
-
-
-
- @Override
- public void execute() {
- machine.fist();
- }
-
-
- }
- package command.example;
- /**
- *
- *作者:alaric
- *时间:2013-8-21上午7:42:21
- *描述:脚踢命令
- */
- public class KickCommand implements Command {
- private GameMachine machine;
-
- public KickCommand(GameMachine machine) {
- super();
- this.machine = machine;
- }
-
- @Override
- public void execute() {
- machine.kick();
- }
-
- }
- package command.example;
- /**
- *
- *作者:alaric
- *时间:2013-8-21上午7:17:02
- *描述:移动命令
- */
- public class MoveCommand implements Command {
- private GameMachine machine;
- private String direction;
-
-
- public MoveCommand(GameMachine machine,String direction) {
- super();
- this.machine = machine;
- this.direction = direction;
- }
-
-
-
- @Override
- public void execute() {
- machine.move(direction);
- }
-
-
-
- }
- package command.example;
- /**
- *
- *作者:alaric
- *时间:2013-8-21上午7:43:14
- *描述:控制类
- */
- public class Control {
-
- private Command fistCommand;
- private Command kickCommand;
- private Command moveCommand;
-
- public Control(Command fistCommand, Command kickCommand, Command moveCommand) {
- super();
- this.fistCommand = fistCommand;
- this.kickCommand = kickCommand;
- this.moveCommand = moveCommand;
- }
-
- public void fist(){
- fistCommand.execute();
- }
-
- public void kick(){
- kickCommand.execute();
- }
-
- public void move(){
- moveCommand.execute();
- }
-
- }
- package command.example;
- /**
- *
- *作者:alaric
- *时间:2013-8-20下午9:26:42
- *描述:客户端角色
- */
- public class Client {
-
- /**
- *作者:alaric
- *时间:2013-8-20下午9:26:36
- *描述:测试
- */
- public static void main(String[] args) {
-
- GameMachine machine = new GameMachine();
- Command fistCommand = new FistCommand(machine);
- Command kickCommand = new KickCommand(machine);
- Command moveCommand = new MoveCommand(machine, "左");
-
- Control control = new Control(fistCommand, kickCommand, moveCommand);
- control.fist();
- control.kick();
- control.move();
-
- //其实在不同命令模式的情况下就是下面这样直接调用,
- //就会让调用者和实际命令执行者紧紧耦合在一起,还有一个好处
- //就是可以在
- //machine.fist();
- //machine.kick();
- //machine.move("左");
- }
-
- }
运行结果如下:
出拳
出脚
向左移动
通过上面代码可以看出,本来客户端可以直接调用接收者来执行动作的,现在在中间引入了命令,这些命令由调用者(Invoker这里是Control)来调用,从而对客户端和命令接收者解耦了,增加了命令后,使得命令除了execute方法外,可以插入很多其它动作,比如redo,undo,或者记录日志等。
设计模式系列目录: