文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

面试必问|哪些场景下Spring的事务会失效?

2024-12-02 13:57

关注

在日常工作中,如果对Spring的事务管理功能使用不当,则会造成Spring事务不生效的问题。而针对Spring事务不生效的问题,也是在跳槽面试中被问的比较频繁的一个问题。

今天,我们就一起梳理下有哪些场景会导致Spring事务生效。

注:部分内容引用自冰河与猫大人出版的《深入理解分布式事务:原理与实战》一书。

文章收录于GitHub和Gitee:

GitHub: https://github.com/sunshinelyz/technology-binghe

Gitee: https://gitee.com/binghe001/technology-binghe

Spring事务不生效总览

简单来说,Spring事务会在几种特定的场景下失效,如下图所示。

数据库不支持事务

Spring事务生效的前提是所连接的数据库要支持事务,如果底层的数据库都不支持事务,则Spring的事务肯定会失效。例如,如果使用的数据库为MySQL,并且选用了MyISAM存储引擎,则Spring的事务就会失效。

事务方法未被Spring管理

如果事务方法所在的类没有加载到Spring IOC容器中,也就是说,事务方法所在的类没有被Spring管理,则Spring事务会失效,示例如下。

  1. public class ProductService { 
  2.  @Autowired 
  3.  private ProductDao productDao; 
  4.  
  5.  @Transactional(propagation = Propagation.REQUIRES_NEW) 
  6.  public void updateProductStockCountById(Integer stockCount, Long id){ 
  7.   productDao.updateProductStockCountById(stockCount, id); 
  8.  } 

ProductService类上没有标注@Service注解,Product的实例没有加载到Spring IOC容器中,就会造成updateProductStockCountById()方法的事务在Spring中失效。

方法没有被public修饰

如果事务所在的方法没有被public修饰,此时Spring的事务会失效,例如,如下代码所示。

  1. @Service 
  2. public class ProductService { 
  3.  @Autowired 
  4.  private ProductDao productDao; 
  5.  
  6.  @Transactional(propagation = Propagation.REQUIRES_NEW) 
  7.  private void updateProductStockCountById(Integer stockCount, Long id){ 
  8.   productDao.updateProductStockCountById(stockCount, id); 
  9.  } 

虽然ProductService上标注了@Service注解,同时updateProductStockCountById()方法上标注了@Transactional(propagation = Propagation.REQUIRES_NEW)注解。

但是,由于updateProductStockCountById()方法为内部的私有方法(使用private修饰),那么此时updateProductStockCountById()方法的事务在Spring中会失效。

同一类中方法调用

如果同一个类中的两个方法分别为A和B,方法A上没有添加事务注解,方法B上添加了 @Transactional事务注解,方法A调用方法B,则方法B的事务会失效。例如,如下代码所示。

  1. @Service 
  2. public class OrderService { 
  3.  
  4.  @Autowired 
  5.  private OrderDao orderDao; 
  6.  
  7.  @Autowired 
  8.  private ProductDao productDao; 
  9.  
  10.  public void submitOrder(){ 
  11.   //生成订单 
  12.   Order order = new Order(); 
  13.   long number = Math.abs(new Random().nextInt(500)); 
  14.   order.setId(number); 
  15.   order.setOrderNo("order_" + number); 
  16.   orderDao.saveOrder(order); 
  17.   //减库存 
  18.   this.updateProductStockCountById(1, 1L); 
  19.  } 
  20.  
  21.  @Transactional(propagation = Propagation.REQUIRES_NEW) 
  22.  public void updateProductStockCountById(Integer stockCount, Long id){ 
  23.   productDao.updateProductStockCountById(stockCount, id); 
  24.  } 

submitOrder()方法和updateProductStockCountById()方法都在OrderService类中,submitOrder()方法上没有标注事务注解,updateProductStockCountById()方法上标注了事务注解,submitOrder()方法调用了updateProductStockCountById()方法,此时,updateProductStockCountById()方法的事务在Spring中会失效。

未配置事务管理器

如果在项目中没有配置Spring的事务管理器,即使使用了Spring的事务管理功能,Spring的事务也不会生效。

例如,没有在项目的配置类中配置如下代码。

  1. @Bean 
  2. public PlatformTransactionManager transactionManager(DataSource dataSource) { 
  3.  return new DataSourceTransactionManager(dataSource); 

此时,Spring的事务就会失效。

方法的事务传播类型不支持事务

如果内部方法的事务传播类型为不支持事务的传播类型,则内部方法的事务在Spring中会失效。

例如,如下代码所示。

  1. @Service 
  2. public class OrderService { 
  3.  @Autowired 
  4.  private OrderDao orderDao; 
  5.  @Autowired 
  6.  private ProductDao productDao; 
  7.  
  8.  @Transactional(propagation = Propagation.REQUIRED) 
  9.  public void submitOrder(){ 
  10.   //生成订单 
  11.   Order order = new Order(); 
  12.   long number = Math.abs(new Random().nextInt(500)); 
  13.   order.setId(number); 
  14.   order.setOrderNo("order_" + number); 
  15.   orderDao.saveOrder(order); 
  16.   //减库存 
  17.   this.updateProductStockCountById(1, 1L); 
  18.  } 
  19.  
  20.  @Transactional(propagation = Propagation.NOT_SUPPORTED) 
  21.  public void updateProductStockCountById(Integer stockCount, Long id){ 
  22.   productDao.updateProductStockCountById(stockCount, id); 
  23.  } 

由于updateProductStockCountById()方法的事务传播类型为NOT_SUPPORTED,不支持事务,则updateProductStockCountById()方法的事务会在Spring中失效。

不正确的捕获异常

不正确的捕获异常也会导致Spring的事务失效,示例如下。

  1. @Service 
  2. public class OrderService { 
  3.  @Autowired 
  4.  private OrderDao orderDao; 
  5.  @Autowired 
  6.  private ProductDao productDao; 
  7.  
  8.  
  9.  @Transactional(propagation = Propagation.REQUIRED) 
  10.  public void submitOrder(){ 
  11.   //生成订单 
  12.   Order order = new Order(); 
  13.   long number = Math.abs(new Random().nextInt(500)); 
  14.   order.setId(number); 
  15.   order.setOrderNo("order_" + number); 
  16.   orderDao.saveOrder(order); 
  17.   //减库存 
  18.   this.updateProductStockCountById(1, 1L); 
  19.  } 
  20.  
  21.  @Transactional(propagation = Propagation.REQUIRED) 
  22.  public void updateProductStockCountById(Integer stockCount, Long id){ 
  23.   try{ 
  24.    productDao.updateProductStockCountById(stockCount, id); 
  25.    int i = 1 / 0; 
  26.   }catch(Exception e){ 
  27.    logger.error("扣减库存异常:", e.getMesaage()); 
  28.   } 
  29.  } 

updateProductStockCountById()方法中使用try-catch代码块捕获了异常,即使updateProductStockCountById()方法内部会抛出异常,但也会被catch代码块捕获到,此时updateProductStockCountById()方法的事务会提交而不会回滚,并且submitOrder()方法的事务会提交而不会回滚,这就造成了Spring事务的回滚失效问题。

错误的标注异常类型

如果在@Transactional注解中标注了错误的异常类型,则Spring事务的回滚会失效,示例如下。

  1. @Transactional(propagation = Propagation.REQUIRED) 
  2. public void updateProductStockCountById(Integer stockCount, Long id){ 
  3.  try{ 
  4.   productDao.updateProductStockCountById(stockCount, id); 
  5.  }catch(Exception e){ 
  6.   logger.error("扣减库存异常:", e.getMesaage()); 
  7.   throw new Exception("扣减库存异常"); 
  8.  } 

在updateProductStockCountById()方法中捕获了异常,并且在异常中抛出了Exception类型的异常,此时,updateProductStockCountById()方法事务的回滚会失效。

为何会失效呢?这是因为Spring中对于默认回滚的事务异常类型为RuntimeException,上述代码抛出的是Exception异常。

默认情况下,Spring事务中无法捕获到Exception异常,所以此时updateProductStockCountById()方法事务的回滚会失效。

此时可以手动指定updateProductStockCountById()方法标注的事务异常类型,如下所示。

  1. @Transactional(propagation = Propagation.REQUIRED,rollbackFor = Exception.class) 

 

这里,需要注意的是:Spring事务注解@Transactional中的rollbackFor属性可以指定 Throwable 异常类及其子类。

本文转载自微信公众号「冰河技术」,可以通过以下二维码关注。转载本文请联系冰河技术公众号。

 

来源:冰河技术内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯