在代理模式中,我们创建具有现有对象的对象,以便向外界提供功能接口。
介绍
意图:为其他对象提供一种代理以控制对这个对象的访问。
主要解决:在直接访问对象时带来的问题,比如说:要访问的对象在远程的机器上。在面向对象系统中,有些对象由于某些原因(比如对象创建开销很大,或者某些操作需要安全控制,或者需要进程外的访问),直接访问会给使用者或者系统结构带来很多麻烦,我们可以在访问此对象时加上一个对此对象的访问层。
何时使用:想在访问一个类时做一些控制。
如何实现:增加中间层,实现与被代理类组合。
应用实例:
Windows 里面的快捷方式。
猪八戒去找高翠兰结果是孙悟空变的,可以这样理解:把高翠兰的外貌抽象出来,高翠兰本人和孙悟空都实现了这个接口,猪八戒访问高翠兰的时候看不出来这个是孙悟空,所以说孙悟空是高翠兰代理类。
买火车票不一定在火车站买,也可以去代售点。
一张支票或银行存单是账户中资金的代理。支票在市场交易中用来代替现金,并提供对签发人账号上资金的控制。
spring aop。
注意事项:
- 和适配器模式的区别:适配器模式主要改变所考虑对象的接口,而代理模式不能改变所代理类的接口。
- 和装饰器模式的区别:装饰器模式为了增强功能,而代理模式是为了加以控制。
实现
1 | public interface Image { |
JDK动态代理
- java.lang.reflect.Proxy:生成动态代理类和对象;
- java.lang.reflect.InvocationHandler(处理器接口):可以通过invoke方法实现对真实角色的代理访问。
- 每次通过Proxy生成的代理类对象都要指定对应的处理器对象。
1 | // InvocationHandler接口 |
- proxy表示被代理的对象
- method表示被代理对象的方法
- args表示方法的参数
1 | // Proxy类的newProxyInstance方法 |
- loader表示委托类的类加载器
- interfaces表示委托类实现的接口(这就说明委托类一定要实现了接口才能使用)
- h表示InvocationHandler接口的子实例,也就是动态代理类的实例
接口
1 | public interface Subject { |
真实对象
1 | public class RealSubject implements Subject{ |
处理器对象
1 | import java.lang.reflect.InvocationHandler; |
调用端
1 | import java.lang.reflect.Proxy; |
优化实现
1 | public class CarProxy implements InvocationHandler { |
Cglib动态代理
Cglib动态代理是针对代理的类,动态生成一个子类,然后子类覆盖代理类中的方法,如果是private或是final类修饰的方法,则不会被重写。(方法拦截技术拦截所有父类方法的调用)
CGLIB是一个功能强大,高性能的代码生成包。它为没有实现接口的类提供代理,为JDK的动态代理提供了很好的补充。通常可以使用Java的动态代理创建代理,但当要代理的类没有实现接口或者为了更好的性能,CGLIB是一个好的选择。
CGLIB作为一个开源项目,其代码托管在github,地址为:https://github.com/cglib/cglib
需要代理的类
1 | public class Engineer { |
Cglib代理类
1 | import net.sf.cglib.proxy.Enhancer; |
测试方法
1 | import java.lang.reflect.Method; |
运行结果
1 | ### before invocation |
优化实现
1 | public class CglibProxy implements MethodInterceptor { |