目录
参考文章:
1、https://www.zhihu.com/question/20794107/answer/658139129(讲解静态、动态代理细致入微)
2、关于Cglib:https://zhuanlan.zhihu.com/p/70098824
1、总体认识
1)、代理总体目的:在目标对象的功能基础上,扩展其功能。不改变其本身的功能;
2)、常见的代理模式:静态、动态、Cglib代理。前两者适合实现了接口的目标对象,后者适合没有实现接口的目标对象;
3)、有接口的目标对象代理思路:基于复制 + 扩展的思路,创建一个代理类:为了更好地在原有基础上进行扩展,可以让代理类和目标类实现同一个接口(类似于复制,然后再进行扩展);
4)、基于不同的“复制“方式,分为静态和动态代理:以下主要用于理解记忆,具体代码参考文章。
- 静态如下:
- jdk动态如下:
跳过生成代理类,直接生成代理对象。
关于代理对象和目标对象之家的桥梁 - InvocationHandler对象:
2、两种动态代理的具体原理
1)jdk动态代理
适合实现了接口的对象。
- 实例原理图:
现在想实现代理实现了Caculator接口的对象,增强其add方法的前后与业务无关的逻辑。
主要大致步骤为:
- 实现 InvocationHandler 接口,有一个实现类,且该接口只有一个invoke方法;
- 创建 JDK 动态代理类:创建 JDK 动态代理类实例同样也是使用反射包中的 java.lang.reflect.Proxy 类进行创建。通过调用
Proxy#newProxyInstance
静态方法(传入三个参数,详细参考文章动态代理部分),写出动态代理类的getProxy方法,能够返回一个代理对象实例;- 执行动态代理:生成一个实例,将其作为参数传入动态代理类的getProxy方法,强转为目标对象,调用目标对象的具体方法,就能实现动态代理了。
2)CGLIB 动态代理
适合没有实现接口的对象。CGLIB 动态代理的实现机制是生成目标类的子类,通过调用父类(目标类)的方法实现,在调用父类方法时再代理中进行增强。
大致步骤为:
- 实现 MethodInterceptor 接口:CGLIB 动态代理通过方法拦截的方式,拦截目标对象的方法,然后再实现代理。通过方法拦截接口调用目标类的方法,然后在该被拦截的方法进行增强处理。具体代码;
创建 CGLIB 动态代理类:同样的,写出动态代理类的getProxy方法,能够返回一个代理对象实例。使用 net.sf.cglib.proxy.Enhancer 类进行创建;
- 执行动态代理:同样的。
tips:Enhancer 在使用过程中,常用且有特色功能还有回调过滤器 CallbackFilter 的使用,它在拦截目标对象的方法时,可以有选择性的执行方法拦截,也就是选择被代理方法的增强处理。(详细参考文章回调过滤器 CallbackFilter 部分)