为什么使用命令模式大B:“上边的代码是否看起来很傻呢,本来可以这样简单实现的。”
publicclassClient{
publicstaticvoidmain(String[]args){
Receiverreceiver=newReceiver();
receiver.actionOne();
receiver.actionTwo();
}
}
publicclassReceiver{
publicReceiver(){
//
}
publicvoidactionOne(){
System.out.println(“ActionOnehasbeentaken.”);
}
publicvoidactionTwo(){
System.out.println(“ActionTwohasbeentaken.”);
}
}
大B:“看!多简洁。如果是像上边如此简单的需求,这个才应该是我们的选择,但是有些情况下这样的写法不能解决的,或者说解决起来不好,所以引入命令模式。1、我们须要Client和Receiver同时开发,而且在开发过程中分别须要不停重构,改名。2、如果我们要求Redo,Undo等功能。3、我们须要命令不按照调用执行,而是按照执行时的情况排序,执行。4、开发后期,我们发现必须要log哪些方法执行了,如何在尽量少更改代码的情况下实现,并且渐少重复代码。5、在上边的情况下,我们的接受者有很多,不止一个。”
小A:“当我们遇到这些情况时应该怎样去解决?”
大B:“解决办法:情况一、我们可以定义一个接口,让Receiver实现这个接口,Client按照接口调用。情况二、我们可以让Receiver记住一些状态,例如执行前的自己的状态,用来undo,但自己记录自己的状态实现起来比较混乱,一般都是一个累记录另一个类的状态。情况三、很难实现。情况四、我们须要在每个Action,前后加上log。”情况五、相对好实现,但是再加上这个,是否感觉最终的实现很混乱呢?
大B:“好,我们再来看看命令模式,在命令模式中,我们增加一些过渡的类,这些类就是上边的命名接口和命令实现,这样就很好的解决了情况一、情况二。我们再加入一个Invoker,这样情况三和情况四就比较好解决了。”
如下加入Log和排序后的。
publicclassInvoker{
privateListcmdList=newArrayList();
publicInvoker(){
}
publicadd(Commandmand){
cmdList.add(mand);
}
publicremove(Commandmand){
cmdList.remove(mand);
}
publicvoidaction(){
Commandcmd;
while((cmd=getCmd())!=null){
log(“begin”+cmd.getName());
cmd.execute();
log(“end”+cmd.getName());
}
}
publicCommandgetCmd(){
//按照自定义优先级,排序取出。
}
}
publicclassClient{
publicstaticvoidmain(String[]args){
Receiverreceiver=newReceiver();
CommandmandOne=newConcreteCommandOne(receiver);
CommandmandTwo=newConcreteCommandTwo(receiver);
Invokerinvoker=newInvoker();
invoker.add(mandOne);
invoker.add(mandTwo);
iinvoker.action();
}
}