spring aop不加载原因和动态代理原理分析

首先,spring aop 是使用动态代理实现的,分布是jdk的动态代理和cglib

一、原理区别:

java动态代理是利用反射机制生成一个实现代理接口的匿名类,在调用具体方法前调用InvokeHandler来处理。

而cglib动态代理是利用asm开源包,对代理对象类的class文件加载进来,通过修改其字节码生成子类来处理。

1、如果目标对象实现了接口,默认情况下会采用JDK的动态代理实现AOP 

2、如果目标对象实现了接口,可以强制使用CGLIB实现AOP 

3、如果目标对象没有实现了接口,必须采用CGLIB库,spring会自动在JDK动态代理和CGLIB之间转换

如何强制使用CGLIB实现AOP?

 (1)添加CGLIB库,SPRING_HOME/cglib/*.jar

 (2)在spring配置文件中加入

 

JDK动态代理和CGLIB字节码生成的区别?

 (1)JDK动态代理只能对实现了接口的类生成代理,而不能针对类

 (2)CGLIB是针对类实现代理,主要是对指定的类生成一个子类,覆盖其中的方法

   因为是继承,所以该类或方法最好不要声明成final 

来源于 https://www.cnblogs.com/leifei/p/8263448.html

 

 

但是,如果我的service里有两个方法,分布是A和B,A实现了aop ,而B 没有,但是B方法调用了A, 这时候通过service.B->A 是不会触发动态代理的

 

 

打印class信息可以看出

上面一个是在controller层通过ioc @Autowired注入的service

下面一个是在service里this.A直接方法调用,就没有使用增强,使用的是原生类

从这里看,我一直以为是spring aop 在生成class的时候将class 改变,强行将实现加载到了当前方法里;

后来发现我完全理解错了,如果是这样,就不再动态了,也就完全忽略了动态两个字;

 

cglib是在java在将class转换为字节码的环节在内存里将二进制字节码改变,随时的动态,只要通过spring的生态环,都可以实现;而类内的相互调用,是直接调用,没有通过代理和注入,是不会进行动态代理的;也就不会aop的加载

 

而jdk的动态代理是通过InvokeHandler,也就是反射机制来做的,在适当的时候,帮你反射调用对应的方法来实现;是完全不同的方式;

 

这两篇文章对我帮助很大

https://www.cnblogs.com/qf123/p/8671479.html

https://blog.csdn.net/luanlouis/article/details/24589193

谢谢博主的研究

你可能感兴趣的:(个人)