@Transactional 失效的几种情况

本文详细探讨了Spring框架中关于非public方法、类内部访问、事务引擎限制、异常处理、传播属性、未抛异常、非Spring管理Bean和异步操作等八个关键问题,并提供了相应的解决策略。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

目录

一、非public修饰的方法

二、类内部访问

三、数据库不支持事务

四、异常类型不匹配

五、传播属性设置问题

六、捕获异常未抛出

七、Bean没有纳入Spring容器管理

八、事务方法内启动新线程进行异步操作


一、非public修饰的方法

@Transactional注解只能在在public修饰的方法下使用。

/**
 * 私有方法上的注解,不生效(因私有方法无法生成代理)
 */
@Transactional
private boolean test() {
    //test code
}

二、类内部访问

类内部非直接访问带注解标记的方法 B,而是通过类普通方法 A,然后由 A 调用 B。

@Service
public class Demo {
  public void A() {
    ......
    B();
  }
  
  @Transactional(rollbackFor = Exception.class)
  public void B() {
     ......
  }
}

失效原因可见:@Transactional类内部访问失效原因详解_Ahuuua的博客-优快云博客

三、数据库不支持事务

MySQL中,MyISAM引擎不支持事物,InnoDB 支持事物

四、异常类型不匹配

@Transactional注解默认处理运行时异常,即只有抛出运行时异常时,才会触发事务回滚,否则并不会如:

/**
 * 非运行异常,且没有通过 rollbackFor 指定抛出的异常,不生效
 *
 * @param id
 * @return
 * @throws Exception
 */
@Transactional
public boolean testException(int id) throws Exception {
    //运行代码
    throw new Exception("参数异常");
}

这种情况需指定异常,如: 

@Transactional(rollbackFor = Exception.class)

五、传播属性设置问题

有几种是不走事务执行的,所以也需要额外注意下

事务特点
REQUIRED默认,如果存在事务,则支持当前事务;不存在,则开启一个新事务
SUPPORTS如果存在一个事务,支持当前事务。如果没有事务,则非事务的执行
MANDATORY需要在一个正常的事务内执行,否则抛异常
REQUIRES_NEW不管存不存在事务,都开启一个新事务
NOT_SUPPORTED不管存不存在,都以非事务方式执行,当存在事务时,挂起事务
NEVER非事务方式执行,如果存在事务,则抛异常
NESTED如果不存在事务,则开启一个事务运行;如果存在事务,则运行一个嵌套事务

propagation属性错误
@Transactional默认的事务传播机制是:REQUIRED,若指定成了NOT_SUPPORTED、NEVER事务传播机制,则事物不生效,如:

@Transactional(propagation = Propagation.NOT_SUPPORTED)

六、捕获异常未抛出

@Transactional
public void A(){
	try{
	   ......
	}catch(Exception e){
	   // 未抛异常
	}
}

七、Bean没有纳入Spring容器管理

// 注释调@Component,该类没被Spring管理,事物也是不生效的
public class Demo {
  
  @Transactional(rollbackFor = Exception.class)
  public void A() {
     ......
  }
}

八、事务方法内启动新线程进行异步操作

@Transactional 注解是 Spring 框架中用来管理事务的注解。它可以将方法标记为事务性方法,确保在方法执行期间,如果发生异常或错误,将回滚事务以确保数据的一致性。 但在以下场景下,@Transactional 注解可能会失效: 1. 注解位置不正确:@Transactional 注解必须应用于 public 可访问性的方法上。如果在 private、protected 或 package-private 的方法上使用,@Transactional 注解将失效。 2. 异常被捕获:如果在事务性方法中抛出异常,但异常被捕获并处理,事务将不会回滚。所以,在捕获异常时一定要小心,确保在处理异常时要么回滚事务,要么重新抛出异常。 3. 事务方法内调用另一个事务方法:如果在一个事务性方法中调用另一个事务性方法,那么内部的事务性方法将会独立于外部事务而执行,除非内部事务方法抛出异常。 4. 非 public 的方法调用:当从一个事务性方法中调用一个非 public 的事务性方法时,事务失效。 5. 静态方法调用:静态方法无法被代理,因此 @Transactional 注解无法应用于静态方法。 6. 缺少事务管理器:如果未配置事务管理器,则 @Transactional 注解将无法正常工作。 7. 异步方法:如果方法被 @Async 注解标记为异步执行,则 @Transactional 注解将失效,因为该方法将在另一个线程中执行。 8. 事务方法被代理对象调用:如果事务性方法是通过代理对象调用的,则 @Transactional 注解可能会失效,因为代理对象可能无法传播事务
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

Ahuuua

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

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

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

打赏作者

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

抵扣说明:

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

余额充值