设计模式之代理模式(Proxy)

意图:

为其他对象提供一种代理以控制对这个对象的访问。

 

种类和用途

Proxy模式根据种类不同,效果也不尽相同:

1、远程(Remote)代理:为一个位于不同的地址空间的对象提供一个局域代表对象。这个不同的地址空间可以是在本机器中,也可是在另一台机器中。远程代理又叫做大使(Ambassador)。好处是系统可以将网络的细节隐藏起来,使得客户端不必考虑网络的存在。客户完全可以认为被代理的对象是局域的而不是远程的,而代理对象承担了大部份的网络通讯工作。由于客户可能没有意识到会启动一个耗费时间的远程调用,因此客户没有必要的思想准备。

2、虚拟(Virtual)代理:根据需要创建一个资源消耗较大的对象,使得此对象只在需要时才会被真正创建。使用虚拟代理模式的好处就是代理对象可以在必要的时候才将被代理的对象加载;代理可以对加载的过程加以必要的优化。当一个模块的加载十分耗费资源的情况下,虚拟代理的好处就非常明显。

3、Copy-on-Write代理:虚拟代理的一种。把复制(克隆)拖延到只有在客户端需要时,才真正采取行动。

4、保护(Protector Access)代理:控制对一个对象的访问,如果需要,可以给不同的用户提供不同级别的使用权限。保护代理的好处是它可以在运行时间对用户的有关权限进行检查,然后在核实后决定将调用传递给被代理的对象。

5、Cache代理:为某一个目标操作的结果提供临时的存储空间,以便多个客户端可以共享这些结果。

6、防火墙(Firewall)代理:保护目标,不让恶意用户接近。

7、同步化(Synchronization)代理:使几个用户能够同时使用一个对象而没有冲突。

8、智能引用(SmartReference)代理:当一个对象被引用时,提供一些额外的操作,比如将对此对象调用的次数记录下来等。

在所有种类的代理模式中,虚拟(Virtual)代理、远程(Remote)代理、智能引用代理(SmartReferenceProxy)和保护(Protector Access)代理是最为常见的代理模式。

 

UML图:


代理模式所涉及的角色有:

抽象主题角色(Subject):声明了真实主题和代理主题的共同接口,这样一来在任何使用真实主题的地方都可以使用代理主题。

代理主题(Proxy)角色:代理主题角色内部含有对真是主题的引用,从而可以在任何时候操作真实主题对象;代理主题角色提供一个与真实主题角色相同的接口,以便可以在任何时候都可以替代真实主体;控制真实主题的应用,负责在需要的时候创建真实主题对象(和删除真实主题对象);代理角色通常在将客户端调用传递给真实的主题之前或之后,都要执行某个操作,而不是单纯的将调用传递给真实主题对象。

真实主题角色(RealSubject)角色:定义了代理角色所代表的真实对象。

 

代码实现:

[cpp]  view plain copy
  1. #include <iostream>  
  2. #include <string>  
  3. using namespace std;  
  4.   
  5. class IGiveGift  
  6. {  
  7.     virtual void GiveDolls() = 0;  
  8.     virtual void GiveFlowers() = 0;  
  9.     virtual void GiveChocolate() = 0;  
  10. };  
  11.   
  12. class SchoolGirl  
  13. {  
  14. private:  
  15.     string name;  
  16. public:  
  17.     SchoolGirl(const string &name): name(name){}  
  18.     string GetName() const  
  19.     {  
  20.         return name;  
  21.     }  
  22. };  
  23.   
  24. class Persuit : public IGiveGift  //送礼物类
  25. {  
  26. private:  
  27.     SchoolGirl *mm;  
  28. public:  
  29.     Persuit(SchoolGirl *mm) : mm(mm){}  
  30.     virtual void GiveDolls()  
  31.     {  
  32.         cout<<mm->GetName()<<"送你洋娃娃"<<endl;  
  33.     }  
  34.   
  35.     virtual void GiveFlowers()  
  36.     {  
  37.         cout<<mm->GetName()<<"送你鲜花"<<endl;  
  38.     }  
  39.   
  40.     virtual void GiveChocolate()  
  41.     {  
  42.         cout<<mm->GetName()<<"送你巧克力"<<endl;  
  43.     }  
  44.   
  45. };  
  46. //送礼物这件事用代理来帮你完成
  47. class Proxy : public IGiveGift   //由代理类来帮你想指定的mm送礼物
  48. {   
  49. private:  
  50.     Persuit *gg;  
  51. public:  
  52.     Proxy(SchoolGirl *mm)  
  53.     {  
  54.         gg = new Persuit(mm);  
  55.     }  
  56.     ~Proxy()  
  57.     {  
  58.         delete gg;  
  59.         gg = NULL;  
  60.     }  
  61.   
  62.     virtual void GiveDolls()  
  63.     {  
  64.         gg->GiveDolls();  
  65.     }  
  66.   
  67.     virtual void GiveFlowers()  
  68.     {  
  69.         gg->GiveFlowers();  
  70.     }  
  71.   
  72.     virtual void GiveChocolate()  
  73.     {  
  74.         gg->GiveChocolate();  
  75.     }  
  76.   
  77. };  
  78. int main(int argc, char **argv)  
  79. {  
  80.     SchoolGirl *mm = new SchoolGirl("lsj");  
  81.     Proxy *sps = new Proxy(mm);  
  82.     sps->GiveDolls();  
  83.     sps->GiveChocolate();  
  84.     sps->GiveFlowers();  
  85.   
  86.     system("pause");  
  87.     return 0;  
  88. }  

与其他模式的区别

1)适配器模式Adapter

适配器Adapter为它所适配的对象提供了一个不同的接口。相反,代理提供了与它的实体相同的接口。然而,用于访问保护的代理可能会拒绝执行实体会执行的操作,因此,它的接口实际上可能只是实体接口的一个子集。

2)装饰器模式Decorator

尽管Decorator的实现部分与代理相似,但Decorator的目的不一样。Decorator为对象添加一个或多个功能,而代理则控制对对象的访问。 

总结

在软件系统中,加一个中间层是我们常用的解决方法,这方面Proxy模式给了我们很好的实现。


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值