将一个请求封装为一个对象,从而使你不同的请求对客户进行参数化;对请求排队或记录请求日志,以及支持可撤销的操作。
二、模式解说
Commad模式是一种对象行为模式,它可以对发送者(sender)和接收者(receiver)完全解耦(decoupling)。("发送者" 是请求操作的对象,"接收者" 是接收请求并执行某操作的对象。有了 "解耦",发送者对接收者的接口一无所知。)这里,"请求"(request)这个术语指的是要被执行的命令。Command模式还让我们可以对 "何时" 以及 "如何" 完成请求进行改变。因此,Command模式为我们提供了灵活性和可扩展性。
三、结构图
Command模式结构图如下:
四、怎么使用?
1)定义一个Command接口,接口中有一个统一的方法,这就是将请求/命令封装为对象。
2)定义具体不同命令类ConcreteCommand实现Command接口。
3)定义一个命令的调用角色Invoker。
4)定义一个命令执行状态的接收者Receiver(非必须)。
五、一个例子
class Document{
public void display(){
System.out.println("显示文档内容");
}
public void undo(){
System.out.println("撤销文档内容");
}
public void redo(){
System.out.println("重做文档内容");
}
}
interface DocumentCommand{
public void execute();
}
class DisplayCommand implements DocumentCommand{
private Document document;
public DisplayCommand(Document doc){
document=doc;
}
public void execute() {
document.display();
}
}
class UndoCommand implements DocumentCommand{
private Document document;
public UndoCommand(Document doc){
document=doc;
}
public void execute() {
document.undo();
}
}
class RedoCommand implements DocumentCommand{
private Document document;
public RedoCommand(Document doc){
document=doc;
}
public void execute(){
document.redo();
}
}
class DocumentInvoker{
private DisplayCommand _dcmd;
private UndoCommand _ucmd;
private RedoCommand _rcmd;
public DocumentInvoker(DisplayCommand dcmd,UndoCommand ucmd,RedoCommand rcmd){
this._dcmd=dcmd;
this._ucmd=ucmd;
this._rcmd=rcmd;
}
public void display(){
_dcmd.execute();
}
public void undo(){
_ucmd.execute();
}
public void redo(){
_rcmd.execute();
}
}
public class CommandTest {
public static void main(String[] args) {
Document doc=new Document();
DisplayCommand display=new DisplayCommand(doc);
UndoCommand undo=new UndoCommand(doc);
RedoCommand redo=new RedoCommand(doc);
DocumentInvoker invoker=new DocumentInvoker(display,undo,redo);
invoker.display();
invoker.undo();
invoker.redo();
}
}
六、适用性
1)使用命令模式作为"CallBack"在面向对象系统中的替代。"CallBack"讲的便是先将一个函数登记上,然后在以后调用此函数。
2)需要在不同的时间指定请求、将请求排队。一个命令对象和原先的请求发出者可以有不同的生命期。换言之,原先的请求发出者可能已经不在了,而命令对象本身仍然是活动的。这时命令的接收者可以是在本地,也可以在网络的另外一个地址。命令对象可以在串形化之后传送到另外一台机器上去。
3)系统需要支持命令的撤消(undo)。命令对象可以把状态存储起来,等到客户端需要撤销命令所产生的效果时,可以调用undo()方法,把命令所产生的效果撤销掉。命令对象还可以提供redo()方法,以供客户端在需要时,再重新实施命令效果。
4)如果一个系统要将系统中所有的数据更新到日志里,以便在系统崩溃时,可以根据日志里读回所有的数据更新命令,重新调用Execute()方法一条一条执行这些命令,从而恢复系统在崩溃前所做的数据更新。
七、优缺点
1)优点: Command模式将 "调用操作的对象"(DocumentInvoker)和 "知道如何执行操作的对象" (Document)完全分离开来。这带来了很大的灵活性:发送请求的对象只需要知道如何发送;它不必知道如何完成请求。
2)会导致某些系统有过多的具体命令类。
八、参考
http://www.1-100.org/JSP/13170.htm
http://terrylee.cnblogs.com/archive/2006/07/17/Command_Pattern.html
http://www.jdon.com/designpatterns/command.htm
http://www.cnblogs.com/zhenyulu/articles/69858.html