命令模式(Command)的定義是:用于將一個請求封裝成一個對象,從而使你可用不同的請求對客戶進(jìn)行參數(shù)化;對請求排隊或者記錄請求日志,以及執(zhí)行可撤銷的操作。也就是說改模式旨在將函數(shù)的調(diào)用、請求和操作封裝成一個單一的對象,然后對這個對象進(jìn)行一系列的處理。此外,可以通過調(diào)用實現(xiàn)具體函數(shù)的對象來解耦命令對象與接收對象。
我們來通過車輛購買程序來展示這個模式,首先定義車輛購買的具體操作類:
$(function () {
var CarManager = {
// 請求信息
requestInfo: function (model, id) {
return 'The information for ' + model +
' with ID ' + id + ' is foobar';
},
// 購買汽車
buyVehicle: function (model, id) {
return 'You have successfully purchased Item '
+ id + ', a ' + model;
},
// 組織view
arrangeViewing: function (model, id) {
return 'You have successfully booked a viewing of '
+ model + ' ( ' + id + ' ) ';
}
};
})();
來看一下上述代碼,通過調(diào)用函數(shù)來簡單執(zhí)行 manager 的命令,然而在一些情況下,我們并不想直接調(diào)用對象內(nèi)部的方法。這樣會增加對象與對象間的依賴?,F(xiàn)在我們來擴展一下這個 CarManager 使其能夠接受任何來自包括 model 和 car ID 的 CarManager 對象的處理請求。根據(jù)命令模式的定義,我們希望實現(xiàn)如下這種功能的調(diào)用:
CarManager.execute({ commandType: "buyVehicle", operand1: 'Ford Escort', operand2: '453543' });
根據(jù)這樣的需求,我們可以這樣啦實現(xiàn) CarManager.execute 方法:
CarManager.execute = function (command) {
return CarManager[command.request](command.model, command.carID);
};
改造以后,調(diào)用就簡單多了,如下調(diào)用都可以實現(xiàn)(當(dāng)然有些異常細(xì)節(jié)還是需要再完善一下的):
CarManager.execute({ request: "arrangeViewing", model: 'Ferrari', carID: '145523' });
CarManager.execute({ request: "requestInfo", model: 'Ford Mondeo', carID: '543434' });
CarManager.execute({ request: "requestInfo", model: 'Ford Escort', carID: '543434' });
CarManager.execute({ request: "buyVehicle", model: 'Ford Escort', carID: '543434' });
命令模式比較容易設(shè)計一個命令隊列,在需求的情況下比較容易將命令計入日志,并且允許接受請求的一方?jīng)Q定是否需要調(diào)用,而且可以實現(xiàn)對請求的撤銷和重設(shè),而且由于新增的具體類不影響其他的類,所以很容易實現(xiàn)。
但敏捷開發(fā)原則告訴我們,不要為代碼添加基于猜測的、實際不需要的功能,如果不清楚一個系統(tǒng)是否需要命令模式,一般就不要著急去實現(xiàn)它,事實上,在需求的時通過重構(gòu)實現(xiàn)這個模式并不困難,只有在真正需求如撤銷、恢復(fù)操作等功能時,把原來的代碼重構(gòu)為命令模式才有意義。