設計模式之命令模式學習理解


命令模式

命令模式(Command Pattern)是一種數據驅動的設計模式,它屬於行為型模式。請求以命令的形式包裹在對象中,並傳給調用對象。調用對象尋找可以處理該命令的合適的對象,並把該命令傳給相應的對象,該對象執行命令。

 

介紹

意圖:將一個請求封裝成一個對象,從而使您可以用不同的請求對客戶進行參數化。

主要解決:在軟件系統中,行為請求者與行為實現者通常是一種緊耦合的關系,但某些場合,比如需要對行為進行記錄、撤銷或重做、事務等處理時,這種無法抵御變化的緊耦合的設計就不太合適。

何時使用:在某些場合,比如要對行為進行"記錄、撤銷/重做、事務"等處理,這種無法抵御變化的緊耦合是不合適的。在這種情況下,如何將"行為請求者"與"行為實現者"解耦?將一組行為抽象為對象,可以實現二者之間的松耦合。

如何解決:通過調用者調用接受者執行命令,順序:調用者→接受者→命令。

關鍵代碼:定義三個角色:1、received 真正的命令執行對象 2、Command 3、invoker 使用命令對象的入口

應用實例:struts 1 中的 action 核心控制器 ActionServlet 只有一個,相當於 Invoker,而模型層的類會隨着不同的應用有不同的模型類,相當於具體的 Command。

優點: 1、降低了系統耦合度。 2、新的命令可以很容易添加到系統中去。

缺點:使用命令模式可能會導致某些系統有過多的具體命令類。

使用場景:認為是命令的地方都可以使用命令模式,比如: 1、GUI 中每一個按鈕都是一條命令。 2、模擬 CMD。

注意事項:系統需要支持命令的撤銷(Undo)操作和恢復(Redo)操作,也可以考慮使用命令模式,見命令模式的擴展。

 

假設現在有個創建工單的類,還有個安裝工的類負責接收和安裝工單。

簡單的緊耦合實現

 1 /**
2 * 新建工單類
3 * @author ko
4 *
5 */
6 public class NewlyBuildOrder {
7
8 public void buildTypeAOrder(){
9 System.out.println("build type a order...");
10 }
11
12 public void buildTypeBOrder(){
13 System.out.println("build type b order...");
14 }
15
16 }
 1 /**
2 * 安裝工
3 * 接單並安裝
4 * @author ko
5 *
6 */
7 public class Installer {
8
9 public void receiveInstall(NewlyBuildOrder newlyBuildOrder){
10
11 newlyBuildOrder.buildTypeAOrder();
12 newlyBuildOrder.buildTypeBOrder();
13 newlyBuildOrder.buildTypeAOrder();
14 newlyBuildOrder.buildTypeAOrder();
15
16 }
17
18 }
 1 /**
2 * 測試類
3 * 新建工單
4 * 安裝工領取並安裝工單
5 * @author ko
6 *
7 */
8 public class Test {
9
10 public static void main(String[] args) {
11 NewlyBuildOrder newlyBuildOrder = new NewlyBuildOrder();
12 Installer installer = new Installer();
13 installer.receiveInstall(newlyBuildOrder);
14 }
15 }
build type a order...
build type b order...
build type a order...
build type a order...

這其實是行為請求者和行為實現者的緊耦合,這樣寫有很多弊端,如果安裝工多了,那就會很混亂,沒有日志記錄,另外需要撤銷領取的工單重新領取也不行。

松耦合命令模式實現

 1 // 把行為請求者對行為實現者的命令抽象成類
2 public abstract class Command {
3
4 protected NewlyBuildOrder newlyBuildOrder;
5
6 public Command(NewlyBuildOrder newlyBuildOrder){
7 this.newlyBuildOrder = newlyBuildOrder;
8 }
9
10 // 執行命令
11 public abstract void excuteCommand();
12
13 }
 1 // A類工單命令
2 public class TypeAOrderCommand extends Command {
3
4 public TypeAOrderCommand(NewlyBuildOrder newlyBuildOrder) {
5 super(newlyBuildOrder);
6 // TODO Auto-generated constructor stub
7 }
8
9 @Override
10 public void excuteCommand() {
11 newlyBuildOrder.buildTypeAOrder();
12 }
13
14 @Override
15 public String toString() {
16 return "TypeAOrderCommand";
17 }
18
19 }
 1 // B類工單命令
2 public class TypeBOrderCommand extends Command {
3
4 public TypeBOrderCommand(NewlyBuildOrder newlyBuildOrder) {
5 super(newlyBuildOrder);
6 // TODO Auto-generated constructor stub
7 }
8
9 @Override
10 public void excuteCommand() {
11 newlyBuildOrder.buildTypeBOrder();
12 }
13
14 @Override
15 public String toString() {
16 return "TypeBOrderCommand";
17 }
18
19 }
 1 /**
2 * 工單管理
3 * 介於行為請求者和行為實現者之間的類
4 * @author ko
5 *
6 */
7 public class OrderManage {
8
9 private List<Command> orders = new ArrayList<>();// 存放具體命令的容器
10
11 // 設置訂單
12 // 對於工單管理類,不管安裝工想要的是什么類型的工單,反正都是‘命令’,只管記錄訂單,
13 //
14 public void setOrder(Command command){
15 if (command.toString().equals("TypeAOrderCommand")) {
16 System.out.println("type a order is over, can't provide ...");
17 }else{
18 orders.add(command);
19 // 打印日志
20 System.out.println("LOG add order:"+command.toString()+",time:"+System.currentTimeMillis());
21 }
22 }
23
24 // 撤銷訂單
25 public void revokeOrder(Command command){
26 orders.remove(command);
27 // 打印日志
28 System.out.println("LOG revoke order:"+command.toString()+",time:"+System.currentTimeMillis());
29 }
30
31 public void notifyBuilder(){
32 for (Command command : orders) {
33 command.excuteCommand();
34 }
35 }
36
37 }
 1 /**
2 * 新建工單類
3 * @author ko
4 *
5 */
6 public class NewlyBuildOrder {
7
8 public void buildTypeAOrder(){
9 System.out.println("build type a order...");
10 }
11
12 public void buildTypeBOrder(){
13 System.out.println("build type b order...");
14 }
15
16 }
 1 /**
2 * 測試類
3 * @author ko
4 *
5 */
6 public class Test {
7
8 public static void main(String[] args) {
9 // 初始准備
10 NewlyBuildOrder newlyBuildOrder = new NewlyBuildOrder();
11 Command typeAOrderCommand = new TypeAOrderCommand(newlyBuildOrder);
12 Command typeBOrderCommand = new TypeBOrderCommand(newlyBuildOrder);
13 OrderManage orderManage = new OrderManage();
14
15 // 設置命令
16 orderManage.setOrder(typeAOrderCommand);
17 orderManage.setOrder(typeBOrderCommand);
18
19 // 通知order builder
20 orderManage.notifyBuilder();
21
22 }
23 }
type a order is over, can't provide ...
LOG add order:TypeBOrderCommand,time:1500879972241
build type b order...

  


注意!

本站转载的文章为个人学习借鉴使用,本站对版权不负任何法律责任。如果侵犯了您的隐私权益,请联系我们删除。



 
粤ICP备14056181号  © 2014-2021 ITdaan.com