Spring @Transactional注解事务六大失效场景

82415ba718dee871bed44fc8396718b8.png

@Transactional事务失效场景1:注解在非public修饰的方法上。

原因:Spring强制的要求。

代码示例:

@Transactional
    private void createOrder(){


    }

@Transactional事务失效场景2:注解在被final关键字修饰的方法上。

原因:Spring的@Transactional注解事务是通过生成一个代理子类,通过重写父类方法方式实现事务的代理增强,被final关键字修饰的方法由于不能被子类重写,所以不能通过代理增强。

代码示例:

@Transactional
    public final void createOrder(){
        
    }

@Transactional事务失效场景3:通过this调用方法

原因:这个跟第2个场景类似,由于this这个关键字在jvm中指当前类对象,而不是spring代理后的代理对象,所以不具备事务增强能力。

代理示例:

public void methodA(){
        //methodB方法如果抛出异常,事务不会回滚
        this.methodB();
    }


    @Transactional
    public void methodB(){


    }

@Transactional事务失效场景4:异常被try捕获。

原因:Spring事务默认只会回滚抛出RuntimeException类型异常,如果异常被try,那么事务就不会回滚。

代码示例:

@Transactional
    public void method() {
        try {
            //business logic process
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

@Transactional事务失效场景5:抛出非RuntimeException类型异常且没有指定rollbackFor。

原因:Spring 注解事务在没有手动指定rollbackFor参数的情况下,默认只会回滚抛出RuntimeException类型异常,可以通过手动指定rollbackFor参数来改变这个策略。

代码示例:

//不会回滚
    @Transactional
    public void methodA() throws Exception {
        try {
            //business logic process
        } catch (Exception e) {
            throw new Exception("抛出非RuntimeException类型异常");
        }
    }


    //会回滚
    @Transactional(rollbackFor = Exception.class)
    public void methodB() throws Exception {
        try {
            //business logic process
        } catch (Exception e) {
            throw new Exception("抛出非RuntimeException类型异常");
        }
    }

@Transactional事务失效场景6:新开线程处理。

原因:Spring事务管理器底层是通过ThreadLocal的原理来管理事务的,无法控制跨线程之间的事务一致性。

示例代码:

//线程内部抛出异常,事务均不会回滚
    @Transactional
    public void methodA(){
        //insert into xxx
        new Thread(new Runnable() {
            @Override
            public void run() {
                // insert into xxx
                // update xxx
                // throw new RuntimeException
            }
        });
    }
---------- 正文结束 ----------
长按扫码关注微信公众号

Java软件编程之家
  • 0
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
@Transactional注解Spring框架中用于开启事务注解,但是在某些情况下,@Transactional注解可能会失效,导致事务无法正常工作。以下是一些可能导致@Transactional注解失效场景: 1. 在同一个类中的两个@Transactional方法之间的调用:如果在同一个类中的两个@Transactional方法之间进行调用,那么事务注解将被忽略,因为Spring无法拦截这样的调用。 2. 异常被catch后没有重新抛出:如果在@Transactional方法中捕获了异常并在catch块中处理了它,但是没有重新抛出异常,那么事务将被提交,而不是回滚。 3. 事务方法中使用了try-catch块:如果在@Transactional方法中使用了try-catch块,并且在catch块中处理了异常,那么事务将被提交,而不是回滚。 4. 事务方法中使用了ThreadLocal:如果在@Transactional方法中使用了ThreadLocal,那么事务将被提交,而不是回滚。 5. 事务方法中使用了static方法:如果在@Transactional方法中使用了static方法,那么事务将被提交,而不是回滚。 6. 事务方法中使用了private方法:如果在@Transactional方法中使用了private方法,那么事务将被提交,而不是回滚。 7. 事务方法中使用了同步方法:如果在@Transactional方法中使用了同步方法,那么事务将被提交,而不是回滚。 8. 事务方法中使用了非公共方法:如果在@Transactional方法中使用了非公共方法,那么事务将被提交,而不是回滚。 9. 事务方法中使用了final方法:如果在@Transactional方法中使用了final方法,那么事务将被提交,而不是回滚。 10. 事务方法中使用了接口默认方法:如果在@Transactional方法中使用了接口默认方法,那么事务将被提交,而不是回滚。 11. 事务方法中使用了lambda表达式:如果在@Transactional方法中使用了lambda表达式,那么事务将被提交,而不是回滚。 12. 事务方法中使用了异步方法:如果在@Transactional方法中使用了异步方法,那么事务将被提交,而不是回滚。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值