CGLIB动态代理
原理:利用asm开源包字节码技术,对代理对象类的class文件加载进来,通过修改其字节码生成子类来处理。
优点:JDK动态代理要求被代理的类必须实现接口,当需要代理的类没有实现接口时Cglib代理是一个很好的选择。另一个优点是Cglib动态代理比使用java反射的JDK动态代理要快
缺点:对于被代理类中的final方法,无法进行代理,因为子类中无法重写final函数
什么是CGLIB动态代理
使用cglib[Code Generation Library]实现动态代理,并不要求委托类必须实现接口,底层采用asm字节码生成框架生成代理类的字节码
依赖
<dependency>
<groupId>cglib</groupId>
<artifactId>cglib</artifactId>
<version>3.1</version>
</dependency>
CGLIB动态代理相关代码
实现MethodInterceptor接口的intercept方法后,所有生成的代理方法都调用这个方法。
intercept方法的具体参数有
obj 目标类的实例
1. method 目标方法实例(通过反射获取的目标方法实例)
2. args 目标方法的参数
3. proxy 代理类的实例
该方法的返回值就是目标方法的返回值。
class UserDaoImpl {
public void add() {
System.out.println("add...");
}
}
/**
* <p>Title: CglibProxy</p>
* <p>Description: 使用cglib[Code Generation Library]实现动态代理,并不要求委托类必须实现接口,底层采用asm字节码生成框架生成代理类的字节码 </p>
* @author liuwq
* @date 2018年11月30日 下午5:40:25
*/
public class CglibProxy implements MethodInterceptor {
private Object target;
public Object getInstance(Object target) {
this.target = target;
// 操作字节码,生成虚拟子类
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(target.getClass());
enhancer.setCallback(this);
return enhancer.create();
}
@Override
public Object intercept(Object obj, java.lang.reflect.Method method, Object[] args, MethodProxy proxy) throws Throwable {
System.out.println("before...");
Object invoke = method.invoke(target, args);
System.out.println("after...");
return invoke;
}
/**
* 日志控制层打印log,AOP技术 环绕通知,获取之前和之后进行拦截参数打印
* 1.CGLIB 没有接口依赖关系 字节码ASM技术
* 2.JDK使用反射技术
*/
public static void main(String[] args) {
CglibProxy cglibProxy = new CglibProxy();
UserDaoImpl userDaoImpl = (UserDaoImpl) cglibProxy.getInstance(new UserDaoImpl());
userDaoImpl.add();
}
}
结果
before...
add...
after...
静态代理与动态代理区别
静态代理需要自己写代理类,而动态代理不需要写代理类。
JDK动态代理与CGLIB实现区别
JDK动态代理底层实现:
JDK的动态代理使用Java的反射技术生成动态代理类,只能代理实现了接口的类, 没有实现接口的类不能实现动态代理。
CGLIB动态代理底层实现:
运行时动态的生成一个被代理类的子类(通过ASM字节码处理框架实现),子类重写了被代理类中所有非final的方法,在子类中采用方法拦截的技术拦截所有父类方法的调用,不需要被代理类对象实现接口,从而CGLIB动态代理效率比Jdk动态代理反射技术效率要高。