JDK动态代理与Cglib代理的深度分析
一.JDK动态代理原理
JDK动态代理首先根据需要代理类的接口、类加载以及handler来生成动态代理类:
public static Object newProxyInstance(ClassLoader loader,Class<?>[] interfaces,InvocationHandler h)
其中的最主要逻辑是:
//生成字节码
byte[] proxyClassFile = ProxyGenerator.generateProxyClass(proxyName, interfaces, accessFlags);
根据提供的代理类名称、接口以及修饰符来生成代理类的字节码。
代理类的字节码继承了Proxy类以及实现了需要代理类的接口,提供了局部变量为InvocationHandler的构造函数,对于每一个需要动态代理的方法中都有:
this.h.invoke(this, m1, new Object[] { paramObject });
参数为生成的代理类、当前代理的方法以及方法的参数,通过调用InvocationHandler中的invoke方法来达到增强方法的目的。
因此每次JDK代理类创建的开销为生成新的字节码,这个字节码类继承Proxy类并且实现了接口。
每次JDK代理类运行的开销需要每次通过invoke方法中的反射来调用需要被代理类中的方法,因此每次运行都有反射的开销。
二.Cglib代理原理
Enhancer 创建代理对象的大概步骤如下:
1.生成代理类 Class 二进制字节码。
2.通过 Class.forname() 加载字节码文件, 生成 Class 对象。
3.通过反射机制获得实例构造, 并创建代理类对象。
代理类继承了被代理类并且实现了Factory接口,代理类会给委托方法直接生成两个方法,一个是重写的原方法(代理类重写了委托类中的方法),另外一个是CGLIB$ $方法,会直接调用父类中的原方法。
调用时只要直接运行生成的代码即可,不需要通过反射。
三.区别
1.JDK动态代理针对接口的代理,CGLIB针对类生成子类来完成动态代理。
2.CGLIB调用效率高,JDK动态代理生成代码效率高。