spring对应java web开发的同学来说,都不陌生,其中事务@Transactional在service层更是常常使用。
aop类内部调用不拦截原因
细心的同学也许早就发现当service中的某个没标注@Transactional的方法调用另一个标注了@Transactional的方法时,居然没开启事务。例如
我们知道@Transactional其实就是一个aop代理,是一个cglib动态代理(常用的动态代理有cglib动态代理,有jdk动态代理)。
直接调用insert01(User u)时,其实流程是这样的:
controller或其他service调用insert02(User u)时,流程是这样的:
此处的this指向目标对象,因此调用this.b()将不会执行b事务切面,即不会执行事务增强。
我们通过debugger方式看看controller中的userService和insert02(xxx)中的this,会发现controller中的userService是一个代理对象
aop类内部调用拦截生效的解决方案
方案一--从beanFactory中获取对象
刚刚上面说到controller中的UserService是代理对象,它是从beanFactory中得来的,那么service类内调用其他方法时,也先从beanFacotry中拿出来就OK了。
方案二--获取代理对象
如果aop是使用注解的话,那需要@EnableAspectJAutoProxy(exposeProxy=true,proxyTargetClass=true),如果是xml配置的,把expose-proxy设置为true,如
方案三--将项目转为aspectJ项目
将项目转为aspectJ项目,aop转为aspect 类
方案四--BeanPostProcessor
通过BeanPostProcessor 在目标对象中注入代理对象,定义InjectBeanSelfProcessor类,实现BeanPostProcessor。