@Transactional注解未生效,没有回滚,事务失效的几种场景

1.@Transactional 可以作用于接口、接口方法、类以及类方法上。当作用于类上时,该类的所有 public 方法将都具有该类型的事务属性,作用在private/protected 方法不会生效,虽然 @Transactional 注解可以作用于接口、接口方法、类以及类方法上,但是 Spring 建议不要在接口或者接口方法上使用该注解,因为这只有在使用基于接口的代理时它才会生效。另外, @Transactional 注解应该只被应用到 public 方法上,这是由 Spring AOP 的本质决定的。如果你在 protected、private 或者默认可见性的方法上使用 @Transactional 注解,这将被忽略,也不会抛出任何异常。默认情况下,只有来自外部的方法调用才会被AOP代理捕获,也就是,类内部方法调用本类内部的其他方法并不会引起事务行为,即使被调用方法使用@Transactional注解进行修饰。

2.在项目中,@Transactional(rollbackFor=Exception.class),如果类加了这个注解,那么这个类里面的方法抛出异常,就会回滚,数据库里面的数据也会回滚。在@Transactional注解中如果不配置rollbackFor属性,那么事物只会在遇到RuntimeException的时候才会回滚,加上rollbackFor=Exception.class,可以让事物在遇到非运行时异常时也回滚(写代码出现的空指针等异常,会被回滚,文件读写,网络出问题,spring就没法回滚了)

3.对于出现异常的代码中用到try{}catch(),如果在catch 中只是打印异常,没有抛出,事物就不会rollback,框架也不会接收到异常信息

异常类型:
在这里插入图片描述

4、使用spring声明式事务,在同一个类里用非事务的方法调用事务方法,事务也是不生效的。

式例:

public class TestTransactional {

    public void testAddItem() {
        addTransactionalMethod();
    }

    @Transactional
    public void addTransactionalMethod() {
        //数据库操作,用事务实现原子性

    }

即使testAddItem里面调用的addTransactionalMethod是事务方法,TestTransactional.testAddItem 事务是不会生效的,这是因为spring是通过aop代理对象进行事务管理的,通过代理对象拦截目标方法的执行,在方法执行前后添加事务功能。当我们在调用testAddItem 发现它不是事务方法,所以事务未开启,而addTransactionalMethod方法即使是事务方法也不会开启事务,是因为是在testAddItem里面用的,事务方法会被testAddItem覆盖,相当于事务失效。

@Transactional注解失效场景有以下几种情况: 1. 异常被catch并处理: 当一个使用@Transactional注解标记的方法中发生异常时,如果异常被catch并在方法内部进行了处理,那么事务将不会回滚。这是因为Spring默认只会对被捕获的异常进行回滚处理。 2. 事务方法内部调用其他事务方法: 如果一个使用@Transactional注解标记的方法内部调用了另一个使用@Transactional注解标记的方法,而内部方法没有抛出异常,则外部方法的事务将无效。原因是Spring默认使用了基于代理的事务机制,而代理是通过AOP实现的。在同一个类中,使用@Transactional注解标记的方法调用其他使用@Transactional注解标记的方法,事务失效。 3. 基于自调用的事务: 当一个使用@Transactional注解标记的方法内部调用了自身(即自循环),而且没有使用代理的方式进行调用,事务也会失效。这是因为代理是通过AOP实现的,自调用会绕过代理,导致事务无法生效。 4. 异步方法: 在使用Spring的异步方法时,如果在异步方法内部使用了@Transactional注解标记的方法,事务将无效。这是因为异步方法会在一个新的线程中执行,而事务是基于线程的。因此,在异步方法中使用事务注解是无效的。 5. protected或private方法: 当使用@Transactional注解标记的方法是protected或private修饰的时候,事务也会失效。尽管没有报错,但事务并不会起作用。这是一个常见的错误点,需要特别注意。 综上所述,需要注意以上情况,以确保@Transactional注解的正确使用和事务生效。如果遇到上述场景,可以考虑使用其他方式来实现事务控制,如编程式事务管理或通过代理对象调用。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值