为什么内部方法调用会导致@Transactional失效?

在上一篇文章中,我们列举了@Transactional事务失效的几大场景。其中有一个场景是class内部方法之间的调用,如下:

    @Transactional(rollbackFor = Exception.class)
    public Long createNewUser(RegisterRequest request) {
        UserDO userDO = new UserDO();
        userDO.setLoginName(request.getLoginName());
        userDO.setLoginPwd(request.getLoginPwd());
        userDO.setCreateTime(new Date());
        userDO.setUpdateTime(new Date());
        this.save(userDO);
        //这里还可以保存userinfo扩展信息,双表操作,需要事务
        addUserInfo();
        return userDO.getId();
    }
    
    @Transactional(rollbackFor = Exception.class)
    public void addUserInfo() {
        //.....
    }

多年经验的资深程序员都会在这个点上犯错,那这是什么原因呢?接下来我们简单地分析一下。

@Transactional 是由spring aop机制实现的,究其根本,其实是由java 动态代理机制实现的,这种实现方式要想生效,必须是外部调用,因为只有外部调用才会走代理增强的实现。addUserInfo()的调用方式相当于this.addUserInfo(),this是什么大家都知道吧,它指向的是另一个实例,根本就不是代理类。

好在spring对这种自调用也给出了解决方案,那就是自注入(Self Injection),代码如下:

    @Autowire
    private xxService xxService;

    @Transactional(rollbackFor = Exception.class)
    public Long createNewUser(RegisterRequest request) {
        UserDO userDO = new UserDO();
        userDO.setLoginName(request.getLoginName());
        userDO.setLoginPwd(request.getLoginPwd());
        userDO.setCreateTime(new Date());
        userDO.setUpdateTime(new Date());
        this.save(userDO);
      //这样就生效了
        xxService.addUserInfo();
        return userDO.getId();
    }
    
    @Transactional(rollbackFor = Exception.class)
    public void addUserInfo() {
        //.....
    }

这种解决方案不是很优美,但是也只能这样了。这个自调用问题在Spring AOP中广泛存在,本质上是动态代理无法解决的盲区,只有AspectJ这类静态代理才能解决。

  • 0
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值