-
命令模式:一种行为型设计模式
-
应用场景:
在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书中摘抄的描述了 ~
如有错误,欢迎指正
网友评论