欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C++設(shè)計(jì)模式之命令模式

 更新時(shí)間:2014年10月08日 11:22:45   作者:果凍想  
這篇文章主要介紹了C++設(shè)計(jì)模式之命令模式,本文講解了什么是命令模式、命令模式的使用場(chǎng)合等內(nèi)容,并給出了一個(gè)代碼實(shí)例,需要的朋友可以參考下

前言

又要過(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):

復(fù)制代碼 代碼如下:

#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ān)文章

  • C++超詳細(xì)講解運(yùn)算符重載

    C++超詳細(xì)講解運(yùn)算符重載

    本文包括了對(duì)C++類(lèi)的6個(gè)默認(rèn)成員函數(shù)中的賦值運(yùn)算符重載和取地址和const對(duì)象取地址操作符的重載。運(yùn)算符是程序中最最常見(jiàn)的操作,例如對(duì)于內(nèi)置類(lèi)型的賦值我們直接使用=賦值即可,因?yàn)檫@些編譯器已經(jīng)幫我們做好了,但是對(duì)象的賦值呢?能直接賦值嗎
    2022-06-06
  • 解析如何在C語(yǔ)言中調(diào)用shell命令的實(shí)現(xiàn)方法

    解析如何在C語(yǔ)言中調(diào)用shell命令的實(shí)現(xiàn)方法

    本篇文章是對(duì)如何在C語(yǔ)言中調(diào)用shell命令的方法進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下
    2013-05-05
  • C語(yǔ)言實(shí)現(xiàn)排序算法之歸并排序詳解

    C語(yǔ)言實(shí)現(xiàn)排序算法之歸并排序詳解

    這篇文章主要介紹了C語(yǔ)言實(shí)現(xiàn)排序算法之歸并排序,對(duì)歸并排序的原理及實(shí)現(xiàn)過(guò)程做了非常詳細(xì)的解讀,需要的朋友可以參考下
    2014-07-07
  • C++圖解單向鏈表類(lèi)模板和iterator迭代器類(lèi)模版詳解

    C++圖解單向鏈表類(lèi)模板和iterator迭代器類(lèi)模版詳解

    這篇文章主要為大家詳細(xì)介紹了C++圖解單向鏈表類(lèi)模板和iterator迭代器類(lèi)模版,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來(lái)幫助
    2022-02-02
  • C++ Boost Bimap示例詳細(xì)講解

    C++ Boost Bimap示例詳細(xì)講解

    Boost是為C++語(yǔ)言標(biāo)準(zhǔn)庫(kù)提供擴(kuò)展的一些C++程序庫(kù)的總稱(chēng)。Boost庫(kù)是一個(gè)可移植、提供源代碼的C++庫(kù),作為標(biāo)準(zhǔn)庫(kù)的后備,是C++標(biāo)準(zhǔn)化進(jìn)程的開(kāi)發(fā)引擎之一,是為C++語(yǔ)言標(biāo)準(zhǔn)庫(kù)提供擴(kuò)展的一些C++程序庫(kù)的總稱(chēng)
    2022-11-11
  • C++中的最小生成樹(shù)算法超詳細(xì)教程

    C++中的最小生成樹(shù)算法超詳細(xì)教程

    這篇文章主要介紹了C++中的最小生成樹(shù)算法超詳細(xì)教程,最小生成樹(shù)的最著名的算法有兩個(gè), 一個(gè)是Prim算法, 另一個(gè)當(dāng)然就是Kruskal算法, 接下來(lái), 我將盡我所能的介紹這兩個(gè)算法, 也算是對(duì)自己學(xué)習(xí)的一個(gè)回顧吧,需要的朋友可以參考下
    2023-08-08
  • VSCode搭建STM32開(kāi)發(fā)環(huán)境的方法步驟

    VSCode搭建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-07
  • OpenCV獲取視頻的每一幀并保存為.jpg圖片

    OpenCV獲取視頻的每一幀并保存為.jpg圖片

    這篇文章主要為大家詳細(xì)介紹了OpenCV獲取視頻的每一幀,并保存為.jpg圖片,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2019-07-07
  • c++中的前向聲明用法解讀

    c++中的前向聲明用法解讀

    這篇文章主要介紹了c++中的前向聲明用法解讀,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-06-06
  • C++實(shí)現(xiàn)LeetCode(146.近最少使用頁(yè)面置換緩存器)

    C++實(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

最新評(píng)論