一、简介
首先,Spring AOP的底层实现有两种方式:一种是JDK动态代理,另一种是CGLib的方式。
JDK动态代理主要涉及java.lang.reflect包下边的两个类:Proxy和InvocationHandler。其中,InvocationHandler是一个接口,可以通过实现该接口定义横切逻辑,并通过反射机制调用目标类的代码,动态地将横切逻辑和业务逻辑贬值在一起。
JDK动态代理的有一个限制,就是它只能为接口创建代理实例,而对于没有通过接口定义业务方法的类,就要用CGLib创建动态代理了。CGLib可以为一个类创建一个子类,在子类中采用方法拦截的技术拦截所有父类方法的调用并顺势织入横切逻辑。
二、JDK 和 CGLib动态代理区别
1、JDK动态代理具体实现原理:
通过实现InvocationHandlet接口创建自己的调用处理器;
通过为Proxy类指定ClassLoader对象和一组interface来创建动态代理;
通过反射机制获取动态代理类的构造函数,其唯一参数类型就是调用处理器接口类型;
通过构造函数创建动态代理类实例,构造时调用处理器对象作为参数参入;
JDK动态代理是面向接口的代理模式,如果被代理目标没有接口那么Spring也无能为力,Spring通过Java的反射机制生产被代理接口的新的匿名实现类,重写了其中AOP的增强方法。
2、CGLib动态代理:
CGLib是一个强大、高性能的Code生产类库,可以实现运行期动态扩展java类,Spring在运行期间通过 CGlib继承要被动态代理的类,重写父类的方法,实现AOP面向切面编程呢。
3、两者对比:
举例:
OrderService orderService = (OrderService) DubboContext.getInstance().getBeanFactory().getBean("orderService");
这个是动态代理实现的。
OrderServiceImpl orderServiceImpl = (OrderServiceImpl ) DubboContext.getInstance().getBeanFactory().getBean("orderService");
这个是cglib实现的。
JDK动态代理是面向接口的。
CGLib动态代理是通过字节码底层继承要代理类来实现(如果被代理类被final关键字所修饰,那么抱歉会失败)。
4、使用注意:
如果要被代理的对象是个实现类,那么Spring会使用JDK动态代理来完成操作(Spirng默认采用JDK动态代理实现机制);
如果要被代理的对象不是个实现类,那么,Spring会强制使用CGLib来实现动态代理。
三、JDK 和 CGLib动态代理性能对比
在以前的JDK版本中,JDK动态代理性能并不是很高,jdk1.6以后就比cglib高了。