美文网首页程序员设计模式 BY C++C++设计模式
C++ 设计模式 —— 14.命令模式

C++ 设计模式 —— 14.命令模式

作者: JeremyYv | 来源:发表于2019-12-13 16:01 被阅读0次
    • 命令模式:一种行为型设计模式

    • 应用场景:
      在GOF的《设计模式:可复用面向对象软件的基础》一书中对命令模式是这样说的:将一个请求封装为一个对象,从而使你可以用不同的请求对客户进行参数化,将请求排队或记录请求日志,以及支持可撤销的操作。
      这段话的意思我在刚刚了解这个设计模式时没有很懂,可以在看完全文后再回来看一遍。
      这个设计模式的应用很多。
      即将要开始被天天谈论的12306订票网站,每个用户发起订票请求后,该请求不会被立即执行,而是会进入消息队列。同一用户再次发起请求时如果前一请求尚在队列中排队,新的请求会被忽略,避免了恶意刷票,优化了服务器的性能。而何时将队列中的消息取出,统一处理,将由系统其他设置决定。这个队列就是请求的收集者。
      举例:
      我还是以公司里的审批制度举例。
      在部门每月末收集报销单时,一般会指定一个收集者,这个人负责收集部门内所有的报销单,收集完成后统一交给审批者进行审批。
      而且,不同种类的报销单可能由不同的人审批,都由这个收集者分配。
      这个收集者就是上述的队列,他可以对报销单的审批顺序进行排列,也可以对数目和人员进行记录,以及支持撤销报销的操作。

    • 实现方式:
      需要创建三个类,请求类,审批人类和收集者类
      请求类中要通过成员变量指定审批人,审批人要提供审批接口,收集者类中要有容器用来容纳请求,并提供统一处理的方法。


    以下是命令模式的简单代码实现
    #include <iostream>
    #include <vector>
    #include <string>
    #include <algorithm>
    using namespace std;
    
    //请求抽象类
    class Request
    {
    public:
        Request(int iNo):m_iNo(iNo){}
        virtual void Execute() = 0;
        int GetID(){return m_iNo;}
    
    protected:
        int m_iNo;//请求ID
    };
    
    //审核人
    class  Approver
    {
    public:
        Approver(string strName):m_strName(strName){}
        //审核请求,打印信息
        virtual void DealRequest(Request* pRequest)
        {
            cout << "Request " << pRequest->GetID() << " Approved by " << m_strName << endl;
        }
    private:
        string m_strName;//审核人的名字
    };
    
    //假期请求
    class HolidayRequest : public Request
    {
    public:
        //构造时指定请求ID和审核人
        HolidayRequest(int iNO, Approver* pApprover):Request(iNO), m_pApprover(pApprover){}
        //请求被处理
        virtual void Execute()
        {
            m_pApprover->DealRequest(this);
        }
    private:
        Approver* m_pApprover;
    };
    
    //汇总请求者
    class Invoker
    {
    public:
        //收集请求
        void AddReq(Request* pReq)
        {
            m_vecReq.push_back(pReq);
        }
        //统一申请
        void InvokeAllReq()
        {
            for_each(m_vecReq.begin(), m_vecReq.end(),Invoke);
        }
    private:
        static void Invoke(Request* pReq)
        {
            if(pReq)
            {
                pReq->Execute();
            }
        }
        vector<Request*> m_vecReq;
    };
    
    主函数中的使用
    int main()
    {
        //假设有两个经理负责审批,分别叫Mark和Nancy
        Approver* pManagerMark = new Approver("Mark");
        Approver* pManagerNancy = new Approver("Nancy");
    
        //一个请求收集者
        Invoker* pInvoker = new Invoker();
    
        pInvoker->AddReq(new HolidayRequest(1, pManagerMark));
        pInvoker->AddReq(new HolidayRequest(2, pManagerMark));
        pInvoker->AddReq(new HolidayRequest(3, pManagerNancy));
        pInvoker->AddReq(new HolidayRequest(4, pManagerNancy));
    
        //收集所有请求后统一送去审批
        pInvoker->InvokeAllReq();
    
        return 0;
    }
    
    控制台输出结果
    Request 1  Approved by Mark
    Request 2  Approved by Mark
    Request 3  Approved by Nancy
    Request 4  Approved by Nancy
    

    嗯 ~ 现在可以回去再看一遍文章开头从GOF书中摘抄的描述了 ~


    如有错误,欢迎指正

    相关文章

      网友评论

        本文标题:C++ 设计模式 —— 14.命令模式

        本文链接:https://www.haomeiwen.com/subject/nvifnctx.html