【设计模式】代理模式

 

代理模式(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;
}
复制代码

 

 

 

 

嗯,有点像简单工厂模式和代理模式的结合了。

   

 

 

posted on   崔好好  阅读(289)  评论(0编辑  收藏  举报

编辑推荐:
· AI与.NET技术实操系列:基于图像分类模型对图像进行分类
· go语言实现终端里的倒计时
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
阅读排行:
· 25岁的心里话
· 闲置电脑爆改个人服务器(超详细) #公网映射 #Vmware虚拟网络编辑器
· 零经验选手,Compose 一天开发一款小游戏!
· 通过 API 将Deepseek响应流式内容输出到前端
· AI Agent开发,如何调用三方的API Function,是通过提示词来发起调用的吗
历史上的今天:
2013-05-15 【windows核心编程】 第三章 内核对象

导航

< 2025年3月 >
23 24 25 26 27 28 1
2 3 4 5 6 7 8
9 10 11 12 13 14 15
16 17 18 19 20 21 22
23 24 25 26 27 28 29
30 31 1 2 3 4 5
点击右上角即可分享
微信分享提示