您好,登錄后才能下訂單哦!
本篇文章為大家展示了怎么在PHP項目中實現一個命令模式,內容簡明扼要并且容易理解,絕對能使你眼前一亮,通過這篇文章的詳細介紹希望你能有所收獲。
GoF定義:將一個請求封裝為一個對象,從而使你可用不同的請求對客戶進行參數化;對請求排隊或記錄請求日志,以及支持可撤消的操作
GoF類圖
class Invoker { public $command; public function __construct($command) { $this->command = $command; } public function exec() { $this->command->execute(); } }
首先我們定義一個命令的接收者,或者說是命令的請求者更恰當。類圖中的英文定義這個單詞是“祈求者”。也就是由它來發起和操作命令。
abstract class Command { protected $receiver; public function __construct(Receiver $receiver) { $this->receiver = $receiver; } abstract public function execute(); } class ConcreteCommand extends Command { public function execute() { $this->receiver->action(); } }
接下來是命令,也就是我們的“菜單”。這個命令的作用是為了定義真正的執行者是誰。
class Receiver { public $name; public function __construct($name) { $this->name = $name; } public function action() { echo $this->name . '命令執行了!', PHP_EOL; } }
接管者,也就是執行者,真正去執行命令的人。
// 準備執行者 $receiverA = new Receiver('A'); // 準備命令 $command = new ConcreteCommand($receiverA); // 請求者 $invoker = new Invoker($command); $invoker->exec();
客戶端的調用,我們要聯系好執行者也就是挑有好廚子的飯館(Receiver),然后準備好命令也就是菜單(Command),最后交給服務員(Invoker)。
其實這個飯店的例子已經非常清晰了,對于命令模式真是完美的解析
那說好的可以下多份訂單或者給多個廚師呢?別急,下面的代碼幫助我們解決這個問題
完整代碼: https://github.com/zhangyue0503/designpatterns-php/blob/master/09.command/source/command.php
<?php class Invoker { private $command = []; public function setCommand(Command $command) { $this->command[] = $command; } public function exec() { if(count($this->command) > 0){ foreach ($this->command as $command) { $command->execute(); } } } public function undo() { if(count($this->command) > 0){ foreach ($this->command as $command) { $command->undo(); } } } } abstract class Command { protected $receiver; protected $state; protected $name; public function __construct(Receiver $receiver, $name) { $this->receiver = $receiver; $this->name = $name; } abstract public function execute(); } class ConcreteCommand extends Command { public function execute() { if (!$this->state || $this->state == 2) { $this->receiver->action(); $this->state = 1; } else { echo $this->name . '命令正在執行,無法再次執行了!', PHP_EOL; } } public function undo() { if ($this->state == 1) { $this->receiver->undo(); $this->state = 2; } else { echo $this->name . '命令未執行,無法撤銷了!', PHP_EOL; } } } class Receiver { public $name; public function __construct($name) { $this->name = $name; } public function action() { echo $this->name . '命令執行了!', PHP_EOL; } public function undo() { echo $this->name . '命令撤銷了!', PHP_EOL; } } // 準備執行者 $receiverA = new Receiver('A'); $receiverB = new Receiver('B'); $receiverC = new Receiver('C'); // 準備命令 $commandOne = new ConcreteCommand($receiverA, 'A'); $commandTwo = new ConcreteCommand($receiverA, 'B'); $commandThree = new ConcreteCommand($receiverA, 'C'); // 請求者 $invoker = new Invoker(); $invoker->setCommand($commandOne); $invoker->setCommand($commandTwo); $invoker->setCommand($commandThree); $invoker->exec(); $invoker->undo(); // 新加一個單獨的執行者,只執行一個命令 $invokerA = new Invoker(); $invokerA->setCommand($commandOne); $invokerA->exec(); // 命令A已經執行了,再次執行全部的命令執行者,A命令的state判斷無法生效 $invoker->exec();
這一次我們一次性解決了多個訂單、多位廚師的問題,并且還順便解決了如果下錯命令了,進行撤銷的問題
可以看出來,命令模式將調用操作的對象與知道如何實現該操作的對象實現了解耦
這種多命令多執行者的實現,有點像 組合模式 的實現
在這種情況下,增加新的命令,即不會影響執行者,也不會影響客戶。當有新的客戶需要新的命令時,只需要增加命令和請求者即可。即使有修改的需求,也只是修改請求者。
Laravel框架的事件調度機制中,除了觀察者模式外,也很明顯的能看出命令模式的影子
我們的手機工廠和餐廳其實并沒有什么兩樣,當我們需要代工廠來制作手機時,也是先下訂單,這個訂單就可以看做是命令。在這個訂單中,我們會規定好需要用到的配件,什么型號的CPU,什么型號的內存,預裝什么系統之類的。然后代工廠的工人們就會根據這個訂單來進行生產。在這個過程中,我不用關心是某一個工人還是一群工人來執行這個訂單,我只需要將這個訂單交給和我們對接的人就可以了,然后只管等著手機生產出來進行驗收咯!!
完整代碼: https://github.com/zhangyue0503/designpatterns-php/blob/master/09.command/source/command-up.php
短信功能又回來了,我們發現除了工廠模式外,命令模式貌似也是一種不錯的實現方式哦。在這里,我們依然是使用那幾個短信和推送的接口,話不多說,我們用命令模式再來實現一個吧。當然,有興趣的朋友可以接著實現我們的短信撤回功能哈,想想上面的命令取消是怎么實現的。
短信發送類圖
完整源碼: https://github.com/zhangyue0503/designpatterns-php/blob/master/09.command/source/command-message.php
<?php class SendMsg { private $command = []; public function setCommand(Command $command) { $this->command[] = $command; } public function send($msg) { foreach ($this->command as $command) { $command->execute($msg); } } } abstract class Command { protected $receiver = []; public function setReceiver($receiver) { $this->receiver[] = $receiver; } abstract public function execute($msg); } class SendAliYun extends Command { public function execute($msg) { foreach ($this->receiver as $receiver) { $receiver->action($msg); } } } class SendJiGuang extends Command { public function execute($msg) { foreach ($this->receiver as $receiver) { $receiver->action($msg); } } } class SendAliYunMsg { public function action($msg) { echo '【阿X云短信】發送:' . $msg, PHP_EOL; } } class SendAliYunPush { public function action($msg) { echo '【阿X云推送】發送:' . $msg, PHP_EOL; } } class SendJiGuangMsg { public function action($msg) { echo '【極X短信】發送:' . $msg, PHP_EOL; } } class SendJiGuangPush { public function action($msg) { echo '【極X推送】發送:' . $msg, PHP_EOL; } } $aliMsg = new SendAliYunMsg(); $aliPush = new SendAliYunPush(); $jgMsg = new SendJiGuangMsg(); $jgPush = new SendJiGuangPush(); $sendAliYun = new SendAliYun(); $sendAliYun->setReceiver($aliMsg); $sendAliYun->setReceiver($aliPush); $sendJiGuang = new SendJiGuang(); $sendAliYun->setReceiver($jgMsg); $sendAliYun->setReceiver($jgPush); $sendMsg = new SendMsg(); $sendMsg->setCommand($sendAliYun); $sendMsg->setCommand($sendJiGuang); $sendMsg->send('這次要搞個大活動,快來注冊吧!!');
在這個例子中,依然是多命令多執行者的模式
可以將這個例子與抽象工廠進行對比,同樣的功能使用不同的設計模式來實現,但是要注意的是,抽象工廠更多的是為了生產對象返回對象,而命令模式則是一種行為的選擇
我們可以看出命令模式非常適合形成命令隊列,多命令讓命令可以一條一條執行下去
它允許接收的一方決定是否要否決請求,Receiver做為實現者擁有更多的話語權
上述內容就是怎么在PHP項目中實現一個命令模式,你們學到知識或技能了嗎?如果還想學到更多技能或者豐富自己的知識儲備,歡迎關注億速云行業資訊頻道。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。