设计模式与实例代码:Proxy模式
来源:互联网 发布:房地产市场调研 知乎 编辑:程序博客网 时间:2024/05/20 09:45
定义与意图:
为某一对象提供一个代理和占位,以达到控制访问的目的。有三种不同的代理,远程代理,虚拟代理,保护代理。
类图:
对比:
主要是使用场景上容易和装饰器模式混淆。因为我们在代理类中为真正的被代理对象装饰了一定的功能,但从类图结构上,二者有较大的区分度。代理需要转发被代理对象的所有接口。而装饰器类没有这种约束。
在选择上,主要从意图上进行区分。如果目的是接管某一类型的所有操作,控制对其进行访问,那么应该使用代理模式,通常在编译时已经确定。而如果只是对某一类型的特定功能进行动态扩展,则使用装饰器模式。
示例代码
#include <iostream>using namespace std;class Subject{public: void Request();};class RealSubject : Subject{public :void Request(){cout<<"Called RealSubject.Request()"<<endl;}};class Proxy : Subject{private :RealSubject* _realSubject;public: void Request(){// Use 'lazy initialization'if (_realSubject == NULL){_realSubject = new RealSubject();}_realSubject->Request();}};void main(){// Create proxy and request a serviceProxy* proxy = new Proxy();proxy->Request();system("pause");}
Java中的动态代理
从代理的类图来看,代理类要实现被代理类的所有接口,要表现的像一个被代理对象,即便只是想控制一两个接口,也需要实现所有其他接口,这样在实现代码时要比较繁琐。为此java提供了动态代理的实现方式。动态代理会拦截被代理类型上的所有的方法调用,这样可以根据调用名称过滤我们需要特殊处理的方法,其他的转发给原始对象处理即可。动态代理创建方法是Proxy.newInstance()方法,这个方法的第二个参数是要代理的类型的方法数组,第三个参数是代理类的类型。
示例代理:
public interface Subject { public void doSomething1(); public void doSomething2();}class RealSubject implements Subject{ @Override public void doSomething1() { System.out.println("RealSubject doSomething1"); } @Override public void doSomething2() { System.out.println("RealSubject doSomething2"); } }class LogProxy implements InvocationHandler{ private Object subject = null; public LogProxy(Object object){ this.subject = object; } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("before call method:" + method.getName()); Object obj = method.invoke(this.subject, args); System.out.println("after call method:" + method.getName()); return obj; } }public class Client { public static void main(String[] vargs) { Subject subject = (Subject) Proxy.newProxyInstance( RealSubject.class.getClassLoader(), RealSubject.class.getInterfaces(), new LogProxy(new RealSubject())); subject.doSomething1(); subject.doSomething2(); }}
Java中许多框架都使用了动态代理的功能,所谓面向方面的编程AOP也依赖于动态代理进行实现。但动态代理的限制是只能代理实现了特定的接口的类型的对象。如果一个类型没有实现任何接口,那么就不能用动态代理了。为了解决这种限制,新的一些库如cglib等出现,其原理是在生成的java类的字节码中插入代码,以达到动态实现被代理对象的方法委托。由于cglib依赖于java虚拟机字节码的实现,这种字节码在后面的Android系统所使用的dalvik及art虚拟机并不兼容,也就无法在android中进行使用。
不过在android开源项目中也有bytecode的生成工具,如byte Buddy就是一个支持android通过字节码生成类的开源项目。利用此项目,就可以实现在特定的方法中插入代理,从而实现android上任意类型的动态代理。
参考文献:
http://dz.sdut.edu.cn/blog/subaochen/2013/05/java%E5%8A%A8%E6%80%81%E4%BB%A3%E7%90%86%E7%9A%84%E7%90%86%E8%A7%A3/
http://www.deepinmind.com/jvm/2014/07/10/how-my-new-friend-byte-buddy-enables-annotation-driven-java-runtime-code-generation.html
- 设计模式与实例代码:Proxy模式
- 设计模式与实例代码:Facade模式
- 设计模式与实例代码:Adapter模式
- 设计模式与实例代码:Strategy模式
- 设计模式与实例代码:Decorator模式
- 设计模式与实例代码:Bridge模式
- 设计模式与实例代码:Observer模式
- 设计模式实例与代码:Prototype模式
- 设计模式实例与代码:Builder模式
- 设计模式与实例代码:Composite模式
- 设计模式与实例代码:Visitor模式
- proxy与stub设计模式
- 设计模式----Proxy模式
- 设计模式:Proxy模式
- 设计模式 - Proxy模式
- 设计模式--Proxy 模式
- 设计模式与实例代码:单例模式
- 设计模式与实例代码:Template Method模式
- Android Studio目录结构及工程项目结构解析
- Android 嵌套滑动机制(NestedScrolling)
- Xcode7 中创建静态库:.a 和 .framework(二)
- 遗传网络算法
- Crazy Calendar Light OJ 1393 (Nim博弈)
- 设计模式与实例代码:Proxy模式
- 正则表达式
- 学习mycat过程回顾
- 06-图2 Saving James Bond - Easy Version (25分)
- Java基本语法和命名规范
- QT + vs 运行的项目出现控制台的解决办法
- windows下安装SVN服务器端和客户端和MyEclipse支持
- XCODE7以后不能使用HTTP协议解决办法
- Intellij IDEA15:建立Scala的Maven项目