总结一下Spring中事务失效的8中场景~

阿Q说代码 2023-11-13 13:58 发表于山东

以下文章来源于JAVA日知录 ,作者飘渺Jam

JAVA日知录.

写代码的架构师,做架构的程序员! 实战、源码、数据库、架构...只要你来,你想了解的这里都有!

大家好,今天来聊一个老生常谈的话题,即Spring中事务失效有哪些场景。我这里整理归纳了一下,大概在以下8种场景下事务会不生效。

1. 数据库引擎不支持事务

这里以 MySQL为例,MyISAM引擎是不支持事务操作的,一般要支持事务都会使用InnoDB引擎,根据MySQL 的官方文档说明,从MySQL 5.5.5 开始的默认存储引擎是 InnoDB,之前默认的都是 MyISAM,所以这一点要值得注意,如果底层引擎不支持事务,那么再怎么设置也没有用。

2.没有被 Spring 管理

示例如下:

public class OrderServiceImpl implements OrderService{
  @Transactional
  public void updateOrder(Order order){
    //update order
  }
}

如果此时把@Service注解注释掉,那么这个类就不会被加载成一个Bean,这个类就不会Spring管理了,事务自然就失效了。

3. 方法不是 public 的

@Transactional注解只能用干public 的方法上,否则事多不会生效,如果要用在非public的方法上,则可以开启基于 AspcetJ 框架的静态代理模式。

4.发生自身调用

示例如下:

@Service
public class OrderServiceImpl implements OrderService {
  public void update(Order order) {
    updateOrder(order);
  }
}

@Transactional
public void updateOrder(0rder order) {
    // update order
  }
}

update 方法上面没有加 @Transactional 注解,如果调用有 @Transactional 注解的updateOrder 方法,那么 updateOrder 方法上的事务还可以生效吗?   这里大家可以先想一想,后面会揭晓答案。

再来看下面这个例子:

@Service
public class OrderServiceImpl implements OrderService {
  @Transactional
  public void update(Order order) {
    updateOrder(order);
  }
}

@Transactional(propagation = Propagation.REQUIRES_NEW)
public void updateOrder(0rder order) {
     updateOrder(order);
  }
}

这次在 update 方法上加了 @Transactional, 如果在 updateOrder 上加了 REOUIRES_NEW新开启一个事务,那么新开启的事务可以生效吗?

这两个例子中的事务都不会生效,因为它们发生了自身调用,就调用了该类自己的方法,而没有经过Spring的代理类,默认只有调用外部代理类的方法,事务才会生效,这也是老生常谈的问题了。

这个问题的解决方案之一就是在事务所在的类中注入自己,用往入的对象再调用另外一个方法,这个不太优雅,在Spring 中可以在当前线程中暴露并获取当前代理类,通过在启动类上添加以下注解来启用暴露代理类,如下面的示例所示。

@EnableAspectJAutoProxy(exposeProxy = true)

然后通过以下代码获取当前代理类,并调用代理类的事务方法:

((0rderService) AopContext.currentProxy()).updateOrder();

Spring 默认只有调用 Spring代理类的public 方法,事务才能生效。

5.没有配置事务管理器

如果没有配置以下DataSourceTransactionManager数据源事务管理器,那么事务也不会生效 :

@Bean
public PlatformTransactionManager transactionManager(DataSource dataSource) {
  return new DataSourceTransactionManager(dataSource);
}  

但在 Spring Boot 中只要引入了 spring-boot-starter-data-jdbc 启动器依赖就会自动配置DataSourceTransactionManager数据源事务管理器,所以 Spring Boot框架不存在这个问题,但在传统的 Spring 框架中需要注意。

6. 设置了不支持事务

示例如下:

@Service
public class OrderServiceImpl implements OrderService {
  @Transactional
  public void update(Order order) {
    updateOrder(order);
  }
  
  @Transactional(propagation = Propagation.NOT_SUPPORTED)
  public void updateOrder(Order order) {
    //update order
  }
}

这里的Propagation.NOT_SUPPORTED表示当前方法不以事务方式运行,当前若存在事务则挂起,这就是主动不支持以事务方式运行了。

7. 异常没有被抛出

示例如下:

@Service
public class OrderServiceImpl implements OrderService {
  @Transactional
  public void update(Order order) {
    try{
      // update order
    }catch{
      
    }
  }
}

这个方法把异常给捕获了,但没有抛出来,所以事务不会回滚,只有捕捉到异常事务才会生效。

8. 异常类型不匹配

示例如下:

@Service
public class OrderServiceImpl implements OrderService {
  @Transactional
  public void update(Order order) {
    try{
      // update order
    }catch{
      throw new Exception("更新失败");
    }
  }
}

因为 Spring 默认回滚的是 RuntimeException 异常,和程序抛出的 Exception 异常不匹配,所以事务也是不生效的。如果要触发默认 RuntimeException之外异常的回滚,则需要在 @Transactiona事务注解上指定异常类,示例如下:

@Transactional(rollbackFor = Exception.class)

在今天的文章中总结了使用 @Transactional注解导致事务失效的几个常见场景,如果 @Transactional事务不生效,则可以根据这几种情形排查一下,其实次数最多的也就是发生自身调用、异常被捕获、异常抛出类型不匹配这几种场景。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Spring事务可能会失效的几个场景包括: 1. 未配置事务管理器:如果在项目没有配置Spring事务管理器,即使使用了Spring事务管理功能,事务也不会生效。[2] 2. 数据库不支持事务Spring事务的生效前提是所连接的数据库要支持事务。如果底层的数据库不支持事务,那么Spring事务肯定会失效。举个例子,如果使用的数据库为MySQL,并且选用了MyISAM存储引擎,那么Spring事务就会失效。 3. 事务方法未被Spring管理:另一个导致事务失效场景事务方法未被Spring管理。要使Spring事务生效,需要在应用程序使用@Transactional注解或配置声明式事务的XML配置,来标记需要进行事务管理的方法。如果方法未被正确标记,那么Spring将无法管理该方法的事务,从而导致事务失效总结起来,Spring事务可能失效场景包括未配置事务管理器、数据库不支持事务以及事务方法未被Spring管理。在使用Spring事务管理功能时,需要注意这些场景,以确保事务的正常生效。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* *3* [Spring 事务失效的 8 种场景!](https://blog.csdn.net/sufu1065/article/details/122076645)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 100%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值