溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

PHP中命令模式的示例分析

發(fā)布時(shí)間:2021-07-09 10:15:46 來(lái)源:億速云 閱讀:109 作者:小新 欄目:編程語(yǔ)言

這篇文章主要介紹PHP中命令模式的示例分析,文中介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們一定要看完!

命令模式,也稱為動(dòng)作或者事務(wù)模式,很多教材會(huì)用飯館來(lái)舉例。作為顧客的我們是命令的下達(dá)者,服務(wù)員是這個(gè)命令的接收者,菜單是這個(gè)實(shí)際的命令,而廚師是這個(gè)命令的執(zhí)行者。

那么,這個(gè)模式解決了什么呢?當(dāng)你要修改菜單的時(shí)候,只需要和服務(wù)員說(shuō)就好了,她會(huì)轉(zhuǎn)達(dá)給廚師,也就是說(shuō),我們實(shí)現(xiàn)了顧客和廚師的解耦。也就是調(diào)用者與實(shí)現(xiàn)者的解耦。

當(dāng)然,很多設(shè)計(jì)模式可以做到這一點(diǎn),但是命令模式能夠做到的是讓一個(gè)命令接收者實(shí)現(xiàn)多個(gè)命令(服務(wù)員下單、拿酒水、上菜),或者把一條命令轉(zhuǎn)達(dá)給多個(gè)實(shí)現(xiàn)者(熱菜廚師、涼菜廚師、主食師傅)。這才是命令模式真正發(fā)揮的地方??!

Gof類圖及解釋

GoF定義:將一個(gè)請(qǐng)求封裝為一個(gè)對(duì)象,從而使你可用不同的請(qǐng)求對(duì)客戶進(jìn)行參數(shù)化;對(duì)請(qǐng)求排隊(duì)或記錄請(qǐng)求日志,以及支持可撤消的操作

GoF類圖

PHP中命令模式的示例分析

代碼實(shí)現(xiàn)

class Invoker
{
    public $command;
    
    public function __construct($command)
    {
        $this->command = $command;
    }

    public function exec()
    {
        $this->command->execute();
    }
}

首先我們定義一個(gè)命令的接收者,或者說(shuō)是命令的請(qǐng)求者更恰當(dāng)。類圖中的英文定義這個(gè)單詞是“祈求者”。也就是由它來(lái)發(fā)起和操作命令。

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();
    }
}

接下來(lái)是命令,也就是我們的“菜單”。這個(gè)命令的作用是為了定義真正的執(zhí)行者是誰(shuí)。

class Receiver
{
    public $name;

    public function __construct($name)
    {
        $this->name = $name;
    }

    public function action()
    {
        echo $this->name . '命令執(zhí)行了!', PHP_EOL;
    }
}

接管者,也就是執(zhí)行者,真正去執(zhí)行命令的人。

// 準(zhǔn)備執(zhí)行者
$receiverA = new Receiver('A');

// 準(zhǔn)備命令
$command = new ConcreteCommand($receiverA);

// 請(qǐng)求者
$invoker = new Invoker($command);
$invoker->exec();

客戶端的調(diào)用,我們要聯(lián)系好執(zhí)行者也就是挑有好廚子的飯館(Receiver),然后準(zhǔn)備好命令也就是菜單(Command),最后交給服務(wù)員(Invoker)。

  • 其實(shí)這個(gè)飯店的例子已經(jīng)非常清晰了,對(duì)于命令模式真是完美的解析

  • 那說(shuō)好的可以下多份訂單或者給多個(gè)廚師呢?別急,下面的代碼幫助我們解決這個(gè)問(wèn)題

完整代碼: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 . '命令正在執(zhí)行,無(wú)法再次執(zhí)行了!', PHP_EOL;
        }

    }
    
    public function undo()
    {
        if ($this->state == 1) {
            $this->receiver->undo();
            $this->state = 2;
        } else {
            echo $this->name . '命令未執(zhí)行,無(wú)法撤銷了!', PHP_EOL;
        }
    }
}

class Receiver
{
    public $name;
    public function __construct($name)
    {
        $this->name = $name;
    }
    public function action()
    {
        echo $this->name . '命令執(zhí)行了!', PHP_EOL;
    }
    public function undo()
    {
        echo $this->name . '命令撤銷了!', PHP_EOL;
    }
}

// 準(zhǔn)備執(zhí)行者
$receiverA = new Receiver('A');
$receiverB = new Receiver('B');
$receiverC = new Receiver('C');

// 準(zhǔn)備命令
$commandOne = new ConcreteCommand($receiverA, 'A');
$commandTwo = new ConcreteCommand($receiverA, 'B');
$commandThree = new ConcreteCommand($receiverA, 'C');

// 請(qǐng)求者
$invoker = new Invoker();
$invoker->setCommand($commandOne);
$invoker->setCommand($commandTwo);
$invoker->setCommand($commandThree);
$invoker->exec();
$invoker->undo();

// 新加一個(gè)單獨(dú)的執(zhí)行者,只執(zhí)行一個(gè)命令
$invokerA = new Invoker();
$invokerA->setCommand($commandOne);
$invokerA->exec();

// 命令A(yù)已經(jīng)執(zhí)行了,再次執(zhí)行全部的命令執(zhí)行者,A命令的state判斷無(wú)法生效
$invoker->exec();
  • 這一次我們一次性解決了多個(gè)訂單、多位廚師的問(wèn)題,并且還順便解決了如果下錯(cuò)命令了,進(jìn)行撤銷的問(wèn)題

  • 可以看出來(lái),命令模式將調(diào)用操作的對(duì)象與知道如何實(shí)現(xiàn)該操作的對(duì)象實(shí)現(xiàn)了解耦

  • 這種多命令多執(zhí)行者的實(shí)現(xiàn),有點(diǎn)像組合模式的實(shí)現(xiàn)

  • 在這種情況下,增加新的命令,即不會(huì)影響執(zhí)行者,也不會(huì)影響客戶。當(dāng)有新的客戶需要新的命令時(shí),只需要增加命令和請(qǐng)求者即可。即使有修改的需求,也只是修改請(qǐng)求者。

  • Laravel框架的事件調(diào)度機(jī)制中,除了觀察者模式外,也很明顯的能看出命令模式的影子

我們的手機(jī)工廠和餐廳其實(shí)并沒(méi)有什么兩樣,當(dāng)我們需要代工廠來(lái)制作手機(jī)時(shí),也是先下訂單,這個(gè)訂單就可以看做是命令。在這個(gè)訂單中,我們會(huì)規(guī)定好需要用到的配件,什么型號(hào)的CPU,什么型號(hào)的內(nèi)存,預(yù)裝什么系統(tǒng)之類的。然后代工廠的工人們就會(huì)根據(jù)這個(gè)訂單來(lái)進(jìn)行生產(chǎn)。在這個(gè)過(guò)程中,我不用關(guān)心是某一個(gè)工人還是一群工人來(lái)執(zhí)行這個(gè)訂單,我只需要將這個(gè)訂單交給和我們對(duì)接的人就可以了,然后只管等著手機(jī)生產(chǎn)出來(lái)進(jìn)行驗(yàn)收咯!!

https://github.com/zhangyue0503/designpatterns-php/blob/master/09.command/source/command-up.php

實(shí)例

短信功能又回來(lái)了,我們發(fā)現(xiàn)除了工廠模式外,命令模式貌似也是一種不錯(cuò)的實(shí)現(xiàn)方式哦。在這里,我們依然是使用那幾個(gè)短信和推送的接口,話不多說(shuō),我們用命令模式再來(lái)實(shí)現(xiàn)一個(gè)吧。當(dāng)然,有興趣的朋友可以接著實(shí)現(xiàn)我們的短信撤回功能哈,想想上面的命令取消是怎么實(shí)現(xiàn)的。

短信發(fā)送類圖

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云短信】發(fā)送:' . $msg, PHP_EOL;
    }
}

class SendAliYunPush
{
    public function action($msg)
    {
        echo '【阿X云推送】發(fā)送:' . $msg, PHP_EOL;
    }
}

class SendJiGuangMsg
{
    public function action($msg)
    {
        echo '【極X短信】發(fā)送:' . $msg, PHP_EOL;
    }
}

class SendJiGuangPush
{
    public function action($msg)
    {
        echo '【極X推送】發(fā)送:' . $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('這次要搞個(gè)大活動(dòng),快來(lái)注冊(cè)吧?。?#39;);

說(shuō)明

  • 在這個(gè)例子中,依然是多命令多執(zhí)行者的模式

  • 可以將這個(gè)例子與抽象工廠進(jìn)行對(duì)比,同樣的功能使用不同的設(shè)計(jì)模式來(lái)實(shí)現(xiàn),但是要注意的是,抽象工廠更多的是為了生產(chǎn)對(duì)象返回對(duì)象,而命令模式則是一種行為的選擇

  • 我們可以看出命令模式非常適合形成命令隊(duì)列,多命令讓命令可以一條一條執(zhí)行下去

  • 它允許接收的一方?jīng)Q定是否要否決請(qǐng)求,Receiver做為實(shí)現(xiàn)者擁有更多的話語(yǔ)權(quán)

以上是“PHP中命令模式的示例分析”這篇文章的所有內(nèi)容,感謝各位的閱讀!希望分享的內(nèi)容對(duì)大家有幫助,更多相關(guān)知識(shí),歡迎關(guān)注億速云行業(yè)資訊頻道!

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

php
AI