spring事务注解@Transactional不生效、不回滚原因

本文详细解析了Spring框架中@Transactional注解的使用方法及其参数配置,包括事务传播行为、隔离级别、读写控制等,同时提供了确保事务正确执行的注意事项。

一般情况下在方法上直接添加@Transactional即可,但是极有可能会达不到我们想要的效果,这时就需要添加相应的参数,参数如下。

@Transactional(propagation=Propagation.REQUIRED)           //控制事务传播。默认是Propagation.REQUIRED
@Transactional(isolation=Isolation.DEFAULT)                //控制事务隔离级别。默认跟数据库的默认隔离级别相同
@Transactional(readOnly=false)                             //控制事务可读写还是只可读。默认可读写
@Transactional(timeout=30)                                 //控制事务的超时时间,单位秒。默认跟数据库的事务控制系统相同,又说是30秒
@Transactional(rollbackFor=RuntimeException.class)         //控制事务遇到哪些异常才会回滚。默认是RuntimeException
@Transactional(rollbackForClassName=RuntimeException)      //同上
@Transactional(noRollbackFor=NullPointerException.class)   //控制事务遇到哪些异常不会回滚。默认遇到非RuntimeException不会回滚
@Transactional(noRollbackForClassName=NullPointerException)//同上

除了添加以上参数进行配置外,还有一些注意事项,可能会造成事务打开或回滚失败:

  1. 添加注解的方法不是public修饰的;
  2. 并不是所有的异常都能触发回滚,默认只有这些会触发回滚"运行时、未检查异常(继承自 RuntimeException 的异常)或者 Error",所以如果想要所有的异常都回滚,需要在注解上添加(rollbackFor = Exception.class)这个参数和值。
  3. 事务只有在方法抛出异常时才会回滚,如果方法中的异常被try-catch捕获,则不会回滚,如果业务需要在catch中进行业务逻辑处理,则需要手动回滚,在catch中添加以下语句:TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
  4. 如果在controller层调用service方法(方法A)没有添加@Transactional注解,那么即使方法A调用(有注解的)方法B,也不会触发事务;

 

另外:附上 事务传播行为 和 事务隔离级别 的相关参数以供诸君查阅:


事务传播行为:

  • REQUIRED(默认):业务方法需要在一个容器里运行。如果方法运行时,已经处在一个事务中,那么加入到这个事务,否则自己新建一个新的事务。 
  • NOT_SUPPORTED:声明方法不需要事务。如果方法没有关联到一个事务,容器不会为他开启事务,如果方法在一个事务中被调用,该事务会被挂起,调用结束后,原先的事务会恢复执行。 
  • REQUIRESNEW:不管是否存在事务,该方法总汇为自己发起一个新的事务。如果方法已经运行在一个事务中,则原有事务挂起,新的事务被创建。 
  • MANDATORY:该方法只能在一个已经存在的事务中执行,业务方法不能发起自己的事务。如果在没有事务的环境下被调用,容器抛出例外。 
  • SUPPORTS:该方法在某个事务范围内被调用,则方法成为该事务的一部分。如果方法在该事务范围外被调用,该方法就在没有事务的环境下执行。 
  • NEVER:该方法绝对不能在事务范围内执行。如果在就抛例外。只有该方法没有关联到任何事务,才正常执行。 
  • NESTED:如果一个活动的事务存在,则运行在一个嵌套的事务中。如果没有活动事务,则按REQUIRED属性执行。它使用了一个单独的事务,这个事务拥有多个可以回滚的保存点。内部事务的回滚不会对外部事务造成影响。它只对DataSourceTransactionManager事务管理器起效。 

事务隔离级别:

  • DEFAULT(默认):DEFAULT为数据源(数据库)的默认隔离级别,以目前常用的MySQL为例,默认的隔离级别通常为REPEATABLE_READ。
  • READ_UNCOMMITTED未授权读取级别:这是最低的隔离级别,一个事务能读取到别的事务未提交的更新数据,很不安全,可能出现丢失更新、脏读、不可重复读、幻读。
  • READ_COMMITTED授权读取级别:以操作同一行数据为前提,读事务允许其他读事务和写事务,未提交的写事务禁止其他读事务和写事务。此隔离级别可以防止更新丢失、脏读,但不能防止不可重复读、幻读。此隔离级别可以通过“瞬间共享读锁”和“排他写锁”实现。
  • REPEATABLE_READ可重复读取级别:保证同一事务中先后执行的多次查询将返回同一结果,不受其他事务影响。以操作同一行数据为前提,读事务禁止其他写事务,但允许其他读事务,未提交的写事务禁止其他读事务和写事务。此隔离级别可以防止更新丢失、脏读、不可重复读,但不能防止幻读。
  • SERIALIZABLE序列化级别:所有的事务依次逐个执行,这样事务之间就完全不可能产生干扰。提供严格的事务隔离,此隔离级别可以防止更新丢失、脏读、不可重复读、幻读。如果仅仅通过“行级锁”是无法实现事务序列化的,必须通过其他机制保证新插入的数据不会被刚执行查询操作的事务访问到。

 

### 作用及行为 `@Transactional(readOnly = true)` 是 Spring 框架中用于声明事务只读属性的注解。它主要影响事务的行为,特别是在与数据库交互时,可以优化性能并确保数据一致性。 1. **事务的只读语义** 当一个事务被标记为只读时,它表明该事务会对数据库进行写操作。这种语义可以被数据库系统识别,并据此进行优化,例如避免必要的锁操作或减少事务日志的写入[^4]。 2. **性能优化** 数据库在处理只读事务时,通常可以跳过一些写操作相关的开销,如锁的获取和释放、事务日志的记录等。这可以显著提升查询操作的性能,尤其是在高并发场景下[^1]。 3. **事务传播行为** 如果一个方法被标记为 `@Transactional(readOnly = true)`,并且它被另一个事务方法调用,则只读事务的传播行为会根据事务传播规则(如 `PROPAGATION_REQUIRED`)决定是否加入现有事务或创建新事务。在大多数情况下,只读事务会加入现有的事务上下文[^2]。 4. **事务代理机制** `@Transactional` 注解依赖于 Spring 的 AOP(面向切面编程)机制,通过动态代理(JDK 动态代理或 CGLIB)在运行时为目标方法创建代理对象。代理对象负责在方法执行前后插入事务管理逻辑,例如事务的提交或回滚。对于只读事务,代理逻辑会设置事务的只读标志,以便底层数据库驱动程序进行相应的优化[^3]。 5. **异常处理与回滚** 在只读事务中,如果发生异常,默认情况下事务仍然会回滚。然而,由于只读事务会修改数据,因此通常需要回滚。可以通过 `@Transactional(rollbackFor = Exception.class)` 明确指定需要回滚的异常类型,以避免必要的回滚操作[^5]。 6. **适用场景** - **查询操作**:适用于修改数据库状态的操作,例如查询、读取数据等。 - **高并发读取**:在需要频繁读取数据且需要修改的场景下,使用只读事务可以减少锁竞争,提高系统吞吐量。 - **数据一致性要求高**:对于需要确保数据一致性的读操作,例如报表生成、数据分析等,只读事务可以避免在事务过程中数据被修改,从而保证读取结果的准确性[^4]。 ### 示例代码 以下是一个使用 `@Transactional(readOnly = true)` 的示例: ```java @Service @Slf4j @Transactional(readOnly = true) public class LikeServiceImpl implements LikeService { @Autowired private LikeMapper likeMapper; @Override @Transactional(readOnly = false) public void saveOrDeleteAll() { // 写操作,事务为非只读 } @Override public List<UserLike> getAll() { // 只读操作,事务为只读 return likeMapper.getAll(); } @Transactional(readOnly = false) @Override public void update(UserLike userLike) { // 写操作,事务为非只读 likeMapper.update(userLike); } } ``` 在该示例中,`getAll()` 方法被标记为只读事务,表明它会修改数据库状态。而 `saveOrDeleteAll()` 和 `update()` 方法则使用了非只读事务,因为它们涉及写操作。 ### 注意事项 - **非 public 方法**:`@Transactional` 注解只能应用在 `public` 方法上,因为 Spring事务代理机制依赖于动态代理,只有 `public` 方法才能被正确代理。 - **类内部方法调用**:如果在一个类中调用另一个方法,而调用者未通过代理对象调用,则事务注解可能失效。因此,建议通过注入的代理对象调用事务方法。 - **数据库支持**:某些数据库可能完全支持只读事务的优化,因此需要根据数据库的具体实现进行测试和验证。 ### 使用建议 - **合理选择只读事务**:在需要修改数据的场景中,尽量使用 `@Transactional(readOnly = true)`,以提升性能。 - **明确异常处理**:在只读事务中,确保异常处理逻辑清晰,避免因异常被捕获而导致事务失效。 - **隔离级别与传播行为**:根据业务需求选择合适的隔离级别和传播行为,以平衡数据一致性和性能。 ---
评论 2
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值