命令(Command)模式,又称Action模式、Tansaction模式。
命令模式(Command Pattern)是一种数据驱动的设计模式,它属于行为型模式。请求以命令的形式包裹在对象中,并传给调用对象。调用对象寻找可以处理该命令的合适的对象,并把该命令传给相应的对象,该对象执行命令。
介绍
-
意图:将一个请求封装成一个对象,从而使您可以用不同的请求对客户进行参数化。
-
主要解决:在软件系统中,行为请求者与行为实现者通常是一种紧耦合的关系,但某些场合,比如需要对行为进行记录、撤销或重做、事务等处理时,这种无法抵御变化的紧耦合的设计就不太合适。
-
何时使用:在某些场合,比如要对行为进行"记录、撤销/重做、事务"等处理,这种无法抵御变化的紧耦合是不合适的。在这种情况下,如何将"行为请求者"与"行为实现者"解耦?将一组行为抽象为对象,可以实现二者之间的松耦合。
-
如何解决:通过调用者调用接受者执行命令,顺序:调用者→命令→接受者。
-
关键代码:定义三个角色:
- received 真正的命令执行对象
- Command
- invoker 使用命令对象的入口
-
应用实例:struts 1 中的 action 核心控制器 ActionServlet 只有一个,相当于 Invoker,而模型层的类会随着不同的应用有不同的模型类,相当于具体的 Command。
-
优点: 1、降低了系统耦合度。 2、新的命令可以很容易添加到系统中去。
-
缺点:使用命令模式可能会导致某些系统有过多的具体命令类。
-
使用场景:认为是命令的地方都可以使用命令模式,比如: 1、GUI 中每一个按钮都是一条命令。 2、模拟 CMD。
-
注意事项:系统需要支持命令的撤销(Undo)操作和恢复(Redo)操作,也可以考虑使用命令模式,见命令模式的扩展。
以下是代码示例:
public class Content {
String msg = "hello everybody ";
}
public abstract class Command {
public abstract void doit(); //exec run
public abstract void undo();
}
public class CopyCommand extends Command {
Content c;
public CopyCommand(Content c) {
this.c = c;
}
@Override
public void doit() {
c.msg = c.msg + c.msg;
}
@Override
public void undo() {
c.msg = c.msg.substring(0, c.msg.length()/2);
}
}
public class DeleteCommand extends Command {
Content c;
String deleted; //用来保存被删除去掉的字符串
public DeleteCommand(Content c) {
this.c = c;
}
@Override
public void doit() {
deleted = c.msg.substring(0, 5);
c.msg = c.msg.substring(5, c.msg.length());
}
@Override
public void undo() {
c.msg = deleted + c.msg;
}
}
public class InsertCommand extends Command {
Content c;
String strToInsert = "http://www.mashibing.com";
public InsertCommand(Content c) {
this.c = c;
}
@Override
public void doit() {
c.msg = c.msg + strToInsert;
}
@Override
public void undo() {
c.msg = c.msg.substring(0, c.msg.length() - strToInsert.length());
}
}
这是客户端:
public class Main {
public static void main(String[] args) {
Content c = new Content();
Command insertCommand = new InsertCommand(c);
insertCommand.doit();
System.out.println(c.msg); // hello everybody http://www.mashibing.com
insertCommand.undo();
System.out.println(c.msg); // hello everybody
Command copyCommand = new CopyCommand(c);
copyCommand.doit();
System.out.println(c.msg); // hello everybody hello everybody
copyCommand.undo();
System.out.println(c.msg); // hello everybody
Command deleteCommand = new DeleteCommand(c);
deleteCommand.doit();
System.out.println(c.msg); // everybody
deleteCommand.undo();
System.out.println(c.msg); // hello everybody
List<Command> commands = new ArrayList<>();
commands.add(new InsertCommand(c));
commands.add(new CopyCommand(c));
commands.add(new DeleteCommand(c));
for(Command comm : commands) {
comm.doit();
}
System.out.println(c.msg);
// everybody http://www.mashibing.comhello everybody http://www.mashibing.com
for(int i= commands.size()-1; i>=0; i--) {
commands.get(i).undo();
}
System.out.println(c.msg);
// hello everybody
}
}
运行截图如下: