Java动态代理技术
Jdk Proxy
- 实现原理
使用Java原生的api,通过反射实现 - 逻辑流程
1.Target Class 及相应的 interface(必须)
2.生成代理类:Proxy.newProxyInstance(class loader,interface list,InvocationHandler)内部逻辑: - 由ProxyClassFactory创建(工厂模式),先得到字节数组,再调用defineClass0解析字节流
- 根据Class获取Constructor,反射(cons.newInstance())创建实例
3.生成的代理类:$Proxy0 extends Proxy implements interface
(Java是单继承,JDK要求代理类必须继承Proxy,因此JDK无法以创建(目标类的)子类方式实现代理)
4.关于InvocationHandler:是基类Proxy的成员,包装了Target Class instance,在创建Proxy Class时通过构造方法注入
5.被代理方法的执行:本质上是被Proxy内部的InvocationHandler实例执行,而这个InvocationHandler实例又是由外部自定义传入的!
cglib proxy
- 实现原理
cglib通过创建一个继承目标类的子类,使用asm库动态修改子类的代码来实现 - 逻辑流程
1.Target Class,不要求必须实现某个接口
2.创建具体的class Generator,例如Enhance enhance;
3.为class Generator设置相关参数
(1)父类,enhance.setSuperClass(XX);
(2)拦截器,enhance.setCallback(new MethodInterceptor()); // 具体增强逻辑
(3)设置过滤器,enhance.setCallbackFilter(); // 具体拦截哪些方法的执行
4.创建代理类:enhance.create();
5.代理方法的执行:被interceptor拦截,实际本质上是在Callback所代表的拦截器里面执行,在执行具体方法时采用了FastClass机制,提升了执行效率?
二者区别
方面 | JDK Proxy | cglib |
---|---|---|
代理范围 | 只能对实现了接口的类进行代理 | 对任何类进行代理 |
执行效率 | 创建代理类速度快 | 创建速度稍慢?执行代理类速度快! |
缓存 | 有 | 有 |
生成class数量 | 少 | 多 |
class、实例生成方式 | 反射:constructor.newInstance | 同样是反射 |
代理类如何持久化到磁盘上
1.jdk proxy
System.getProperties().put(“sun.misc.ProxyGenerator.saveGeneratedFiles”, “true”);
2.cglib proxy
System.setProperty(DebuggingClassWriter.DEBUG_LOCATION_PROPERTY, “/Users/jvm_test/”);
下一步
1.cglib缓存
2.cglib fast class机制
3.Spring动态代理实现