C++設(shè)計(jì)模式之命令模式
前言
又要過(guò)年了,又是一個(gè)搶票季;從大學(xué)起,到現(xiàn)在工作,一直都是在外地,離家千里;以前買(mǎi)票,曾經(jīng)也去火車(chē)站通宵排隊(duì)買(mǎi)票;直到12306的騰空出現(xiàn),在電腦前不停止的點(diǎn)著鼠標(biāo)刷票,那個(gè)時(shí)候12306很是脆弱,搶一張票更是難上加難;現(xiàn)在好了,慢慢強(qiáng)大的12306,買(mǎi)票時(shí)出現(xiàn)了一個(gè)排隊(duì)系統(tǒng),先買(mǎi)票,進(jìn)入12306的排隊(duì)系統(tǒng);然后,系統(tǒng)一個(gè)一個(gè)的處理大家的請(qǐng)求,一旦你的購(gòu)票請(qǐng)求進(jìn)入了排隊(duì)系統(tǒng),你就無(wú)法再次進(jìn)行刷票了,除非你退出排隊(duì)系統(tǒng);這就減少了購(gòu)票者的刷票次數(shù);減少了12306后臺(tái)服務(wù)器的處理壓力。那么,你有沒(méi)有想過(guò),12306是如何將你的購(gòu)票請(qǐng)求加入排隊(duì)系統(tǒng)的呢?這樣的排隊(duì)系統(tǒng)是如何實(shí)現(xiàn)的呢?而我今天總結(jié)的命令模式,將會(huì)對(duì)此進(jìn)行簡(jiǎn)單的剖析。
什么是命令模式?
在GOF的《設(shè)計(jì)模式:可復(fù)用面向?qū)ο筌浖幕A(chǔ)》一書(shū)中對(duì)命令模式是這樣說(shuō)的:將一個(gè)請(qǐng)求封裝為一個(gè)對(duì)象,從而使你可用不同的請(qǐng)求對(duì)客戶(hù)進(jìn)行參數(shù)化;對(duì)請(qǐng)求排隊(duì)或記錄請(qǐng)求日志,以及支持可撤銷(xiāo)的操作。在OOP中,一切都是對(duì)象,將請(qǐng)求封裝成對(duì)象,符合OOP的設(shè)計(jì)思想,當(dāng)將客戶(hù)的單個(gè)請(qǐng)求封裝成對(duì)象以后,我們就可以對(duì)這個(gè)請(qǐng)求存儲(chǔ)更多的信息,使請(qǐng)求擁有更多的能力;命令模式同樣能夠把請(qǐng)求發(fā)送者和接收者解耦,使得命令發(fā)送者不用去關(guān)心請(qǐng)求將以何種方式被處理。
我們?cè)?2306上,單擊購(gòu)票,這是一個(gè)請(qǐng)求,12306將這個(gè)請(qǐng)求封裝為一個(gè)對(duì)象,在12306還沒(méi)有上線排隊(duì)系統(tǒng)時(shí),你買(mǎi)票是這樣的:你不停的用鼠標(biāo)點(diǎn)擊12306網(wǎng)站上的購(gòu)票按鈕,直到你買(mǎi)到了票;對(duì)于你的每一次點(diǎn)擊,服務(wù)器都要進(jìn)行處理,做出響應(yīng),告訴你,有沒(méi)有買(mǎi)到票;這樣,可能就會(huì)出現(xiàn)很多次無(wú)效的點(diǎn)擊,但是這些無(wú)效的點(diǎn)擊卻增加了服務(wù)器的負(fù)擔(dān)。增加了排隊(duì)系統(tǒng)以后,你的購(gòu)票請(qǐng)求就進(jìn)入了對(duì)應(yīng)的購(gòu)票隊(duì)列,一旦你進(jìn)入了購(gòu)票隊(duì)列,當(dāng)你再次鼠標(biāo)單擊購(gòu)票時(shí),12306會(huì)拒絕你的購(gòu)票請(qǐng)求,它會(huì)告訴你,你已經(jīng)進(jìn)入了購(gòu)票隊(duì)列;處于購(gòu)票隊(duì)列中的你,你可以選擇退出購(gòu)票隊(duì)列去購(gòu)買(mǎi)其它車(chē)次的車(chē)票,從而進(jìn)入其它購(gòu)票隊(duì)列。這樣就有效的減少了購(gòu)票者發(fā)送很多無(wú)效的購(gòu)票請(qǐng)求。
這就好比票是共享資源,誰(shuí)都想要,但是票的數(shù)量是一定的;在沒(méi)有排隊(duì)系統(tǒng)之前,大家的購(gòu)票請(qǐng)求都是去競(jìng)爭(zhēng)這個(gè)票,服務(wù)器對(duì)于大家對(duì)于共享資源——票的競(jìng)爭(zhēng)進(jìn)行互斥,誰(shuí)搶到了,票就少一張;而現(xiàn)在有了購(gòu)票隊(duì)列以后,大家都不用去競(jìng)爭(zhēng)了,按時(shí)間的先后順序排好隊(duì),12306把票一張張的發(fā)給進(jìn)入隊(duì)列的購(gòu)票者。
UML類(lèi)圖
Command:聲明執(zhí)行操作的接口;
ConcreteCommand:將一個(gè)接收者對(duì)象綁定于一個(gè)動(dòng)作,之后,調(diào)用接收者相應(yīng)的操作,以實(shí)現(xiàn)Execute來(lái)完成相應(yīng)的命令;
Client:創(chuàng)建一個(gè)具體命令對(duì)象,但是并沒(méi)有設(shè)定它的接收者;
Invoker:要求該命令執(zhí)行這個(gè)請(qǐng)求;
Receiver:知道如何實(shí)施與執(zhí)行一個(gè)請(qǐng)求相關(guān)的操作,任何類(lèi)都可能作為一個(gè)接收者。
以上這些對(duì)象是按照下面的方式進(jìn)行協(xié)作的:
1.Client創(chuàng)建一個(gè)ConcreteCommand命令對(duì)象,并指定它的Receiver對(duì)象;
2.Invoker對(duì)象存儲(chǔ)該ConcreteCommand對(duì)象;
3.該Invoker通過(guò)調(diào)用Command對(duì)象的Execute操作來(lái)提交一個(gè)請(qǐng)求。如果這個(gè)命令請(qǐng)求是可以撤銷(xiāo)的,ConcreteCommand就執(zhí)行Execute操作之前存儲(chǔ)當(dāng)前狀態(tài)以用于取消該命令請(qǐng)求;
4.ConcreteCommand對(duì)象調(diào)用Receiver的一些操作以執(zhí)行該請(qǐng)求。
使用場(chǎng)合
使用命令模式實(shí)現(xiàn)12306(工程下載):
CHomePage類(lèi),表示12306的官網(wǎng)訂票頁(yè)面;
C12306Processor類(lèi),是后臺(tái)真正處理用戶(hù)的請(qǐng)求的類(lèi),專(zhuān)門(mén)進(jìn)行出票;
Command類(lèi),表示用戶(hù)的購(gòu)票命令請(qǐng)求;
Customer類(lèi),表示購(gòu)票的用戶(hù)。
由于代碼較多,這里只提供工程的下載。
這里再提供命令模式的一般實(shí)現(xiàn):
#include <iostream>
using namespace std;
#define SAFE_DELETE(p) if (p) { delete p; p = NULL; }
class Receiver
{
public:
void Action()
{
cout<<"Receiver->Action"<<endl;
}
};
class Command
{
public:
virtual void Execute() = 0;
};
class ConcreteCommand : public Command
{
public:
ConcreteCommand(Receiver *pReceiver) : m_pReceiver(pReceiver){}
void Execute()
{
m_pReceiver->Action();
}
private:
Receiver *m_pReceiver;
};
class Invoker
{
public:
Invoker(Command *pCommand) : m_pCommand(pCommand){}
void Invoke()
{
m_pCommand->Execute();
}
private:
Command *m_pCommand;
};
int main()
{
Receiver *pReceiver = new Receiver();
Command *pCommand = new ConcreteCommand(pReceiver);
Invoker *pInvoker = new Invoker(pCommand);
pInvoker->Invoke();
SAFE_DELETE(pInvoker);
SAFE_DELETE(pCommand);
SAFE_DELETE(pReceiver);
return 0;
}
總結(jié)
命令模式是一個(gè)很經(jīng)典的模式,我的理解也不會(huì)很到位;在我們的身邊,就存在很多的使用命令模式的例子,數(shù)據(jù)庫(kù)中的事務(wù)就是使用命令模式去實(shí)現(xiàn)的,在C#中的委托也是使用命令模式去實(shí)現(xiàn)的。我在這里只是將我在學(xué)習(xí)過(guò)程中理解到的東西記錄了下來(lái)和大家分享??赡苡械牡胤轿业睦斫庖泊嬖诓铄e(cuò),希望大家和我分享你對(duì)命令模式的理解。
- 通過(guò)c++11改進(jìn)我們的模式之改進(jìn)命令模式
- C++設(shè)計(jì)模式編程中使用Bridge橋接模式的完全攻略
- C++設(shè)計(jì)模式編程中的迭代器模式應(yīng)用解析
- 深入剖析設(shè)計(jì)模式中的組合模式應(yīng)用及在C++中的實(shí)現(xiàn)
- 設(shè)計(jì)模式中的備忘錄模式解析及相關(guān)C++實(shí)例應(yīng)用
- 深入解析設(shè)計(jì)模式中的適配器模式在C++中的運(yùn)用
- 詳解state狀態(tài)模式及在C++設(shè)計(jì)模式編程中的使用實(shí)例
- 詳解C++設(shè)計(jì)模式編程中對(duì)狀態(tài)模式的運(yùn)用
- C++設(shè)計(jì)模式編程中Template Method模板方法模式的運(yùn)用
- 詳解設(shè)計(jì)模式中的Command命令模式及相關(guān)C++實(shí)現(xiàn)
相關(guān)文章
解析如何在C語(yǔ)言中調(diào)用shell命令的實(shí)現(xiàn)方法
本篇文章是對(duì)如何在C語(yǔ)言中調(diào)用shell命令的方法進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-05-05C語(yǔ)言實(shí)現(xiàn)排序算法之歸并排序詳解
這篇文章主要介紹了C語(yǔ)言實(shí)現(xiàn)排序算法之歸并排序,對(duì)歸并排序的原理及實(shí)現(xiàn)過(guò)程做了非常詳細(xì)的解讀,需要的朋友可以參考下2014-07-07C++圖解單向鏈表類(lèi)模板和iterator迭代器類(lèi)模版詳解
這篇文章主要為大家詳細(xì)介紹了C++圖解單向鏈表類(lèi)模板和iterator迭代器類(lèi)模版,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來(lái)幫助2022-02-02VSCode搭建STM32開(kāi)發(fā)環(huán)境的方法步驟
當(dāng)我們的工程文件比較大的時(shí)候,編譯一次代碼需要很久可能會(huì)花費(fèi)到四五分鐘,但是我們用vscode編寫(xiě)和編譯的話時(shí)間就會(huì)大大縮減,本文就介紹一下VSCode搭建STM32開(kāi)發(fā)環(huán)境,感興趣的可以了解一下2021-07-07C++實(shí)現(xiàn)LeetCode(146.近最少使用頁(yè)面置換緩存器)
這篇文章主要介紹了C++實(shí)現(xiàn)LeetCode(146.近最少使用頁(yè)面置換緩存器),本篇文章通過(guò)簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-07-07