Latest web development tutorials

命令模式

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

介紹

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

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

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

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

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

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

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

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

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

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

實現

我們首先創建作為命令的接口Order ,然後創建作為請求的Stock類。 實體命令類BuyStockSellStock ,實現了Order接口,將執行實際的命令處理。 創建作為調用對象的類Broker ,它接受訂單並能下訂單。

Broker對象使用命令模式,基於命令的類型確定哪個對象執行哪個命令。 CommandPatternDemo,我們的演示類使用Broker類來演示命令模式。

命令模式的 UML 圖

步驟1

創建一個命令接口。

Order.java

public interface Order {
   void execute();
}

步驟2

創建一個請求類。

Stock.java

public class Stock {
	
   private String name = "ABC";
   private int quantity = 10;

   public void buy(){
      System.out.println("Stock [ Name: "+name+", 
         Quantity: " + quantity +" ] bought");
   }
   public void sell(){
      System.out.println("Stock [ Name: "+name+", 
         Quantity: " + quantity +" ] sold");
   }
}

步驟3

創建實現了Order接口的實體類。

BuyStock.java

public class BuyStock implements Order {
   private Stock abcStock;

   public BuyStock(Stock abcStock){
      this.abcStock = abcStock;
   }

   public void execute() {
      abcStock.buy();
   }
}

SellStock.java

public class SellStock implements Order {
   private Stock abcStock;

   public SellStock(Stock abcStock){
      this.abcStock = abcStock;
   }

   public void execute() {
      abcStock.sell();
   }
}

步驟4

創建命令調用類。

Broker.java

import java.util.ArrayList;
import java.util.List;

   public class Broker {
   private List<Order> orderList = new ArrayList<Order>(); 

   public void takeOrder(Order order){
      orderList.add(order);		
   }

   public void placeOrders(){
      for (Order order : orderList) {
         order.execute();
      }
      orderList.clear();
   }
}

步驟5

使用Broker 類來接受並執行命令。

CommandPatternDemo.java

public class CommandPatternDemo {
   public static void main(String[] args) {
      Stock abcStock = new Stock();

      BuyStock buyStockOrder = new BuyStock(abcStock);
      SellStock sellStockOrder = new SellStock(abcStock);

      Broker broker = new Broker();
      broker.takeOrder(buyStockOrder);
      broker.takeOrder(sellStockOrder);

      broker.placeOrders();
   }
}

步驟6

驗證輸出。

Stock [ Name: ABC, Quantity: 10 ] bought
Stock [ Name: ABC, Quantity: 10 ] sold