一、理解@Transactional底层的AOP原理
最初调用的是AOP代理对象而不是目标对象。执行事务切面,事务切面内部通过TransactionInterceptor环绕增强进行事务的增强,即进入目标方法之前开启事务,退出目标方法时提交/回滚事务。
public interface AService {
public void a();
public void b();
}
@Service()
public class AServiceImpl implements AService{
@Transactional(propagation = Propagation.REQUIRED)
public void a() {
this.b();
}
@Transactional(propagation = Propagation.REQUIRES_NEW)
public void b() {
}
}
二、同类方法的调用无法实现切面中的增强
此处this指向目标对象,因此调用this.b()将不会执行b事务切面,即不会执行事务增强,因此b方法的事务定义“@Transactional(propagation = Propagation.REQUIRES_NEW)”将不会实施。即结果是b和a的事务定义是一样的(可以看到事务切面只对a方法进行了事务增强,没有对b方法进行增强)。
- b中的事务会不会生效?
不会,a的事务会生效,b中不会有事务,因为a中调用b属于内部调用,没有通过代理,所以不会有事务产生。 - 如果想要b中有事务存在,要如何做?
此处a方法中调用b方法时,只要通过AOP代理调用b方法即可执行事务切面,进行事务增强。
三、暴露 AOP 代理
- 常规项目
<aop:aspectj-autoproxy expose-proxy=“true”>
设置expose-proxy属性为true - SpringBoot项目,在启动类上加
@EnableAspectJAutoProxy(exposeProxy = true)
就可以了。
更改后:
使用(AService)AopContext.currentProxy();
获取当前代理,将this.b()改为((AService)AopContext.currentProxy()).b()
。
四、三种判断AOP代理对象的方法
- AopUtils.isAopProxy(bean):是否是代理对象。
- AopUtils.isJdkDynamicProxy(bean):是否是JDK动态代理对象。
- AopUtils.isCglibProxy(bean):是否是CGLIB动态代理对象。