同类方法调用 @Transactional 失效与解决

一、理解@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方法进行增强)。

  1. b中的事务会不会生效?
    不会,a的事务会生效,b中不会有事务,因为a中调用b属于内部调用,没有通过代理,所以不会有事务产生。
  2. 如果想要b中有事务存在,要如何做?
    此处a方法中调用b方法时,只要通过AOP代理调用b方法即可执行事务切面,进行事务增强。

三、暴露 AOP 代理

  1. 常规项目
    <aop:aspectj-autoproxy expose-proxy=“true”>设置expose-proxy属性为true
  2. SpringBoot项目,在启动类上加@EnableAspectJAutoProxy(exposeProxy = true)就可以了。

更改后:
使用(AService)AopContext.currentProxy();获取当前代理,将this.b()改为((AService)AopContext.currentProxy()).b()

四、三种判断AOP代理对象的方法

  1. AopUtils.isAopProxy(bean):是否是代理对象。
  2. AopUtils.isJdkDynamicProxy(bean):是否是JDK动态代理对象。
  3. AopUtils.isCglibProxy(bean):是否是CGLIB动态代理对象。
  • 0
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

JFS_Study

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值