首页 > 代码库 > 【设计模式】代理模式
【设计模式】代理模式
代理模式(Proxy)
关于代理,ITer做常见的应该就是用来翻鸡爱抚大不留的代理了。
生活中也有很多代理,比如在北京不收中介费的在租房中介,这帮人就叫代理,链家这样的房产中介也叫代理,只不过是收中介费的代理。
恰好今天中午在公司楼道里吃午饭(热干面,楼下超市买的),刚吃了几口,进来一个抽烟的哥们,进来跟我打招呼,应该是隔壁公司的,隔壁公司是家律所,他说他不是律师,是做专利的,也就是代理专利,一般介于客户和知识产权局和专利局之间,所以,他就是一个代理。
用下图来说明
代理模式的示意图如下
Subject是一些需要做的工作,只有一些接口,以专利来说,可以有:【提交申请】【提交材料】【交费】等;
RealSubject是真正去做事的人,RealSubject实现了Subject的接口
Proxy是代理,Proxy也实现了Subject的接口,其在这些接口的内部去调用RealSubject实现的相应接口。
代理模式隐藏了真正干活的人RealSubject, 由于提供给客户的是Proxy,所以RealSubject和Proxy应该是相关的。
下面以客户请专利公司代为申请专利为例子来描述,如下
申请专利需要的【申请专利】抽象类,接口
/************************************************************************
设计模式4
代理模式
代理模式,即客户和代理打交道,而代理又在内部调动相应的真正干活的人去干活
对客户来说,真正干活的人是透明的,他只和代理打交道。
以客户通过专利代理公司申请专利为例子来描述代理模式
************************************************************************/
//【申请专利】接口,声明了一组申请专利的操作
class IApplyZL
{
public:
IApplyZL();
virtual ~IApplyZL();
virtual void CommitApply() = 0; //提交申请
virtual void CommitFiles() = 0; //提交资料
virtual void Pay() = 0; //支付费用
};
IApplyZL::IApplyZL(){}
IApplyZL::~IApplyZL(){}
void IApplyZL::CommitApply() {}
void IApplyZL::CommitFiles() {}
void IApplyZL::Pay() {}
真正干活的【代理人/员工】类,员工类继承自IApplyZL抽象基类,实现了所有虚函数,表示员工可以干这个活
//【代理人】类,真正干活的人,他实现(有这样的能力)了申请专利要做的一些工作
class Worker : public IApplyZL
{
public:
Worker(string strClient);
virtual ~Worker();
virtual void CommitApply();
virtual void CommitFiles();
virtual void Pay();
protected:
string m_strClient; //客户名
};
Worker::Worker(string strClient) : m_strClient(strClient){}
Worker::~Worker(){}
void Worker::CommitApply()
{
cout << "\r\n我正在为 "<<m_strClient<<" 先生/女士 提交专利申请!\r\n";
}
void Worker::CommitFiles()
{
cout << "\r\n我正在为 "<<m_strClient<<" 先生/女士 提交专利申请需要的材料!\r\n";
}
void Worker::Pay()
{
cout << "\r\n我正在为 "<<m_strClient<<" 先生/女士 提交专利申请需要需要的费用!\r\n";
}
面向客户的【代理公司】类,他也继承自IApplyZL抽象基类,实现了所有虚函数,表示可以承接这样的业务
//【代理公司】类,客户跟代理公司打交道,代理公司实现了IApplyZL的接口,
//表示代理公司可以承接这样的业务
class ProxyZL : public IApplyZL
{
public:
ProxyZL(string strClient);
virtual ~ProxyZL();
virtual void CommitApply();
virtual void CommitFiles();
virtual void Pay();
protected:
Worker* m_pWorker; //干活的员工
string m_strClient; //客户名
};
ProxyZL::ProxyZL(string strClient) : m_strClient(strClient), m_pWorker(NULL) {}
ProxyZL::~ProxyZL(){}
void ProxyZL::CommitApply()
{
if(NULL == m_pWorker) m_pWorker = new Worker(m_strClient);
m_pWorker->CommitApply();
}
void ProxyZL::CommitFiles()
{
if(NULL == m_pWorker) m_pWorker = new Worker(m_strClient);
m_pWorker->CommitFiles();
}
void ProxyZL::Pay()
{
if(NULL == m_pWorker) m_pWorker = new Worker(m_strClient);
m_pWorker->Pay();
}
main函数以及执行情况
int _tmain(int argc, _TCHAR* argv[])
{
ProxyZL proxy("cuish");
proxy.CommitApply();
proxy.CommitFiles();
proxy.Pay();
cout<<endl<<endl;
return 0;
}
PS:
想到一个问题,总是感觉代理模式和简单工厂模式,策略模式很相似,可是相似在什么地方又不好说。假如在【ProxyZL】类中的成员Worker *m_pWorker改为某个继承自IApplyZL抽象基类的子类的子类,如下:
即class Workers : public IApplyZL{};
ProxyZL类的成员改为Workers *m_pSomeWorker; //某个员工
又有Workers类的若干个子类,Workers类可以继续为抽象基类,在ProxyZL类中根据构造函数的参数不同可以用Workers类的不同子类的实例(new 子类())的指针给m_pSomeWorker指针赋值,再根据多态的性质,即可去调用不同的子类去执行申请专利的操作。 如果是这样,那就和简单工厂/策略模式很类似了。
如下:
//【员工】类,仍然为抽象基类,没有实现IApplyZL中的纯虚函数
class Workers : public IApplyZL
{
public:
Workers(string strClient);
virtual ~Workers();
protected:
string m_strClient; //客户名
};
Workers::Workers(string strClient) : m_strClient(strClient){}
Workers::~Workers(){}
两个员工类
//【代理人1】类,真正干活的人,他实现(有这样的能力)了申请专利要做的一些工作
class Worker1 : public Workers
{
public:
Worker1(string strClient);
virtual ~Worker1();
virtual void CommitApply();
virtual void CommitFiles();
virtual void Pay();
protected:
const string m_strWorkerName;
};
Worker1::Worker1(string strClient) : Workers(strClient), m_strWorkerName(string("worker1")){}
Worker1::~Worker1(){}
void Worker1::CommitApply()
{
cout << "\r\n"<<m_strWorkerName<<"正在为 "<<m_strClient<<" 先生/女士 提交专利申请!\r\n";
}
void Worker1::CommitFiles()
{
cout << "\r\n"<<m_strWorkerName<<"正在为 "<<m_strClient<<" 先生/女士 提交专利申请需要的材料!\r\n";
}
void Worker1::Pay()
{
cout << "\r\n"<<m_strWorkerName<<"正在为 "<<m_strClient<<" 先生/女士 提交专利申请需要需要的费用!\r\n";
}
//////////////////////////////////////////////////////////////////////////
//【代理人2】类,真正干活的人,他实现(有这样的能力)了申请专利要做的一些工作
class Worker2 : public Workers
{
public:
Worker2(string strClient);
virtual ~Worker2();
virtual void CommitApply();
virtual void CommitFiles();
virtual void Pay();
protected:
const string m_strWorkerName; //客户名
};
Worker2::Worker2(string strClient) : Workers(strClient), m_strWorkerName(string("worker2")){}
Worker2::~Worker2(){}
void Worker2::CommitApply()
{
cout << "\r\n"<<m_strWorkerName<<"正在为 "<<m_strClient<<" 先生/女士 提交专利申请!\r\n";
}
void Worker2::CommitFiles()
{
cout << "\r\n"<<m_strWorkerName<<"我正在为 "<<m_strClient<<" 先生/女士 提交专利申请需要的材料!\r\n";
}
void Worker2::Pay()
{
cout << "\r\n"<<m_strWorkerName<<"我正在为 "<<m_strClient<<" 先生/女士 提交专利申请需要需要的费用!\r\n";
}
代理公司类
//【代理公司】类,客户跟代理公司打交道,代理公司实现了IApplyZL的接口,
//表示代理公司可以承接这样的业务
class ProxyZL : public IApplyZL
{
public:
ProxyZL(string strClient);
virtual ~ProxyZL();
virtual void CommitApply();
virtual void CommitFiles();
virtual void Pay();
protected:
Workers* m_pSomeWorker; //某个干活的员工
string m_strClient; //客户名
};
ProxyZL::ProxyZL(string strClient) : m_strClient(strClient)
{
if (m_strClient == string("cuish"))
{
m_pSomeWorker = new Worker1(m_strClient);
}
else if(m_strClient == string("zhangsan"))
{
m_pSomeWorker = new Worker2(m_strClient);
}
else
{
m_pSomeWorker = NULL;
}
}
ProxyZL::~ProxyZL()
{
if (NULL != m_pSomeWorker) delete m_pSomeWorker;
}
void ProxyZL::CommitApply()
{
if(NULL != m_pSomeWorker)
m_pSomeWorker->CommitApply();
}
void ProxyZL::CommitFiles()
{
if(NULL != m_pSomeWorker)
m_pSomeWorker->CommitFiles();
}
void ProxyZL::Pay()
{
if(NULL != m_pSomeWorker)
m_pSomeWorker->Pay();
}
客户端和执行结果
int _tmain(int argc, _TCHAR* argv[])
{
ProxyZL proxy("cuish");
proxy.CommitApply();
proxy.CommitFiles();
proxy.Pay();
cout<<endl<<endl;
ProxyZL proxy2("zhangsan");
proxy2.CommitApply();
proxy2.CommitFiles();
proxy2.Pay();
cout<<endl<<endl;
return 0;
}
嗯,有点像简单工厂模式和代理模式的结合了。