在软件开发过程中,有些对象有时候会由于网络或其他的障碍,以至于不能够或者不能直接访问到这些对象,如果直接访问对象给系统带来不必要的复杂性,这时候可以在客户端和目标对象之间增加一层中间层,让代理对象代替目标对象,然后客户端只需要访问代理对象,由代理对象去帮我们去请求目标对象并返回结果给客户端。
代理模式按照使用目的可以分为以下几种:
- 远程(Remote)代理:为一个位于不同的地址空间的对象提供一个局域代表对象。这个不同的地址空间可以是本电脑中,也可以在另一台电脑中。最典型的例子就是——客户端调用Web服务或WCF服务。
- 虚拟(Virtual)代理:根据需要创建一个资源消耗较大的对象,使得对象只在需要时才会被真正创建。
- Copy-on-Write代理:虚拟代理的一种,把复制(或者叫克隆)拖延到只有在客户端需要时,才真正采取行动。
- 保护(Protect or Access)代理:控制一个对象的访问,可以给不同的用户提供不同级别的使用权限。
- 防火墙(Firewall)代理:保护目标不让恶意用户接近。
- 智能引用(Smart Reference)代理:当一个对象被引用时,提供一些额外的操作,比如将对此对象调用的次数记录下来等。
- Cache代理:为某一个目标操作的结果提供临时的存储空间,以便多个客户端可以这些结果。
代理模式——就是给某一个对象提供一个代理,并由代理对象控制对原对象的引用。在一些情况下,一个客户不想或者不能直接引用一个对象,而代理对象可以在客户端和目标对象之间起到中介的作用。例如电脑桌面的快捷方式就是一个代理对象,快捷方式是它所引用的程序的一个代理。
(1) AccessValidator:身份验证类,业务类,它提供方法Validate()来实现身份验证
1 /AccessValidator.cs 2 using System; 3 4 namespace ProxySample 5 { 6 class AccessValidator 7 { 8 //模拟实现登录验证 9 public bool Validate(string userId) 10 { 11 Console.WriteLine("在数据库中验证用户'" + userId + "'是否是合法用户?"); 12 if (userId.Equals("杨过")) { 13 Console.WriteLine("'{0}'登录成功!",userId); 14 return true; 15 } 16 else { 17 Console.WriteLine("'{0}'登录失败!", userId); 18 return false; 19 } 20 } 21 } 22 }
(2) Logger:日志记录类,业务类,它提供方法Log()来保存日志。
1 //Logger.cs 2 using System; 3 4 namespace ProxySample 5 { 6 class Logger 7 { 8 //模拟实现日志记录 9 public void Log(string userId) { 10 Console.WriteLine("更新数据库,用户'{0}'查询次数加1!",userId); 11 } 12 } 13 }
(3) Searcher:抽象查询类,充当抽象主题角色,它声明了DoSearch()方法。
1 //Searcher.cs 2 namespace ProxySample 3 { 4 interface Searcher 5 { 6 string DoSearch(string userId, string keyword); 7 } 8 }
(4) RealSearcher:具体查询类,充当真实主题角色,它实现查询功能,提供方法DoSearch()来查询信息。
//RealSearcher.cs using System; namespace ProxySample { class RealSearcher : Searcher { //模拟查询商务信息 public string DoSearch(string userId, string keyword) { Console.WriteLine("用户'{0}'使用关键词'{1}'查询商务信息!",userId,keyword); return "返回具体内容"; } } }
(5) ProxySearcher:代理查询类,充当代理主题角色,它是查询代理,维持了对RealSearcher对象、AccessValidator对象和Logger对象的引用。
1 //ProxySearcher.cs 2 namespace ProxySample 3 { 4 class ProxySearcher : Searcher 5 { 6 private RealSearcher searcher = new RealSearcher(); //维持一个对真实主题的引用 7 private AccessValidator validator; 8 private Logger logger; 9 10 public string DoSearch(string userId, string keyword) 11 { 12 //如果身份验证成功,则执行查询 13 if (this.Validate(userId)) 14 { 15 string result = searcher.DoSearch(userId, keyword); //调用真实主题对象的查询方法 16 this.Log(userId); //记录查询日志 17 return result; //返回查询结果 18 } 19 else 20 { 21 return null; 22 } 23 } 24 25 //创建访问验证对象并调用其Validate()方法实现身份验证 26 public bool Validate(string userId) 27 { 28 validator = new AccessValidator(); 29 return validator.Validate(userId); 30 } 31 32 //创建日志记录对象并调用其Log()方法实现日志记录 33 public void Log(string userId) 34 { 35 logger = new Logger(); 36 logger.Log(userId); 37 } 38 } 39 }
(6) 配置文件App.config,在配置文件中存储了代理主题类类名。
1 <?xml version="1.0" encoding="utf-8" ?> 2 <configuration> 3 <appSettings> 4 <add key="proxy" value="ProxySample.ProxySearcher"/> 5 </appSettings> 6 </configuration>
(7) Program:客户端测试类
1 //Program.cs 2 using System; 3 using System.Configuration; 4 using System.Reflection; 5 6 namespace ProxySample 7 { 8 class Program 9 { 10 static void Main(string[] args) 11 { 12 //读取配置文件 13 string proxy = ConfigurationManager.AppSettings["proxy"]; 14 15 //反射生成对象,针对抽象编程,客户端无须分辨真实主题类和代理类 16 Searcher searcher; 17 searcher = (Searcher)Assembly.Load("ProxySample").CreateInstance(proxy); 18 19 String result = searcher.DoSearch("杨过", "玉女心经"); 20 Console.Read(); 21 } 22 } 23 }
4. 结果及分析
编译并运行程序,输出结果如下:
在数据库中验证用户'杨过'是否是合法用户? '杨过'登录成功! 用户'杨过'使用关键词'玉女心经'查询商务信息! 更新数据库,用户'杨过'查询次数加1! |
本实例是保护代理和智能引用代理的应用实例,在代理类ProxySearcher中实现对真实主题类的权限控制和引用计数,如果需要在访问真实主题时增加新的访问控制机制和新功能,只需增加一个新的代理类,再修改配置文件,在客户端代码中使用新增代理类即可,源代码无须修改,符合开闭原则。
本篇源码学习于 https://blog.csdn.net/lovelion/article/details/8228042