命令模式就是解耦强耦合代码,用户只关心功能的实现,开发者却可以利用命令模式在这之间加一些小动作,比如:撤销命令、命令排队、记录命令日志等
命令模式的本质:封装请求。
命令模式的关键就是把请求封装成为命令对象,然后就可以对这个对象进行一系列的处理了,比如上面讲到的参数化配置、可撤销操作、宏命令、队列请求、日志请求等功能处理。
建议在以下情况时选用命令模式。
如果需要抽象出需要执行的动作,并参数化这些对象,可以选用命令模式。将这些需要执行的动作抽象成为命令,然后实现命令的参数化配置。
如果需要在不同的时刻指定、排列和执行请求,可以选用命令模式。将这些请求封装成为命令对象,然后实现将请求队列化。
如果需要支持取消操作,可以选用命令模式,通过管理命令对象,能很容易地实现命令的恢复和重做功能。
如果需要支持当系统崩溃时,能将系统的操作功能重新执行一遍,可以选用命令模式。将这些操作功能的请求封装成命令对象,然后实现日志命令,就可以在系统恢复以后,通过日志获取命令列表,从而重新执行一遍功能。
在需要事务的系统中,可以选用命令模式。命令模式提供了对事务进行建模的方法。命令模式有一个别名就是Transaction。
命令模式的优点
更松散的耦合
命令模式使得发起命令的对象——客户端,和具体实现命令的对象——接收者对象完全解耦,也就是说发起命令的对象完全不知道具体实现对象是谁,也不知道如何实现。
更动态的控制
命令模式把请求封装起来,可以动态地对它进行参数化、队列化和日志化等操作,从而使得系统更灵活。
很自然的复合命令
命令模式中的命令对象能够很容易地组合成复合命令,也就是前面讲的宏命令,从而使系统操作更简单,功能更强大。
更好的扩展性
由于发起命令的对象和具体的实现完全解耦,因此扩展新的命令就很容易,只需要实现新的命令对象,然后在装配的时候,把具体的实现对象设置到命令对象中,然后就可以使用这个命令对象,已有的实现完全不用变化。
模拟电脑开机,点击机箱开机按钮,调用主板初始化系统,然后用户就能操作了
主板类
/*** @description:主板接口*/
public interface MainBoardApi {/*** 开机*/void open();
}/*** @description:技嘉主板*/
public class JiJiaMainBoard implements MainBoardApi{@Overridepublic void open() {System.out.println("技嘉主板正在开机,请稍后");System.out.println("接通电源.............");System.out.println("设备检查.............");System.out.println("装载系统.............");System.out.println("机器正常运行,请操作....");}
}/*** @description:微星主板*/
public class WeiXinMainBoard implements MainBoardApi{@Overridepublic void open() {System.out.println("微星主板正在开机,请稍后");System.out.println("接通电源.............");System.out.println("设备检查.............");System.out.println("装载系统.............");System.out.println("机器正常运行,请操作....");}
}
开机按钮类
/*** @description:机箱开机按钮*/
public class BoxButton {/*** 点击开机按钮,就开机*/public void boot(int flag){if (1==flag){new JiJiaMainBoard().open();}else {new WeiXinMainBoard().open();}}
}
测试类
/*** @description:最开始耦合写法*/
public class Test1 {public static void main(String[] args) {//点击按钮开机new BoxButton().boot(2);}
}
现在,机箱按钮直接调用主板,是强耦合的关系,很不利于维护
改造上面的耦合写法,主板接口和实现类不变
调用程序类(也就是上面的机箱按钮)
/*** @description:调用程序(机箱开机按钮)*/
public class Invoker {/*** 持有命令对象*/private Command command=null;public void setCommand(Command command) {this.command = command;}/*** 开机*/public void boot(){command.execute();}
}
命令类
/*** @description:命令接口*/
public interface Command {/*** 执行命令*/void execute();
}/*** @description:具体命令类(这里是开机命令)*/
@AllArgsConstructor
public class ConcreteCommand implements Command{/*** 持有主板对象*/private MainBoardApi mainBoardApi;@Overridepublic void execute() {//命令类不能进行开机操作//调用主板进行开机mainBoardApi.open();}
}
测试类
/*** @description:测试类* @createTime 2022/11/30 13:06*/
public class Client {public static void main(String[] args) {//把命令和实现组装起来Command command=new ConcreteCommand(new WeiXinMainBoard());//为机箱按钮设置命令Invoker invoker = new Invoker();invoker.setCommand(command);//模拟开机按钮invoker.boot();}
}
效果
撤销有两种:
模拟假如存在一个存在一个撤销按钮,电脑开机后,点击撤销按钮,撤销开机操作,也就是进行关机
主板类增加关机功能
/*** @description:主板接口*/
public interface MainBoardApi {/*** 开机*/void open();/*** 关机*/void close();
}/*** @description:技嘉主板*/
public class JiJiaMainBoard implements MainBoardApi{@Overridepublic void open() {System.out.println("技嘉主板正在开机,请稍后");System.out.println("接通电源.............");System.out.println("设备检查.............");System.out.println("装载系统.............");System.out.println("机器正常运行,请操作....");}@Overridepublic void close() {System.out.println("技嘉主板正在关机,请稍后");System.out.println("关机成功.............");}
}/*** @description:微星主板*/
public class WeiXinMainBoard implements MainBoardApi{@Overridepublic void open() {System.out.println("微星主板正在开机,请稍后");System.out.println("接通电源.............");System.out.println("设备检查.............");System.out.println("装载系统.............");System.out.println("机器正常运行,请操作....");}@Overridepublic void close() {System.out.println("微星主板正在关机,请稍后");System.out.println("关机成功.............");}
}
命令接口增加撤销命令
/*** @description:命令接口*/
public interface Command {/*** 执行命令*/void execute();/*** 撤销命令*/void undo();
}/*** @description:具体命令类(这里是开机命令)*/
@AllArgsConstructor
public class ConcreteCommand implements Command{/*** 持有主板对象*/private MainBoardApi mainBoardApi;@Overridepublic void execute() {//命令类不能进行开机操作//调用主板进行开机mainBoardApi.open();}@Overridepublic void undo() {//撤销开机,也就是关机mainBoardApi.close();}
}
调用程序增加关机功能
/*** @description:调用程序(机箱开机按钮)*/
public class Invoker {/*** 持有命令对象*/private Command command=null;public void setCommand(Command command) {this.command = command;}/*** 开机*/public void boot(){command.execute();}/*** 关机*/public void shutdown(){command.undo();}
}
测试类
public class Client {public static void main(String[] args) {//把命令和实现组装起来Command command=new ConcreteCommand(new WeiXinMainBoard());//为机箱按钮设置命令Invoker invoker = new Invoker();invoker.setCommand(command);//模拟开机按钮invoker.boot();//模拟点击撤销按钮,电脑关机invoker.shutdown();}
}