您现在的位置是:亿华云 > 系统运维
面试必问|哪些场景下Spring的事务会失效?
亿华云2025-10-09 03:19:57【系统运维】1人已围观
简介在日常工作中,如果对Spring的事务管理功能使用不当,则会造成Spring事务不生效的问题。而针对Spring事务不生效的问题,也是在跳槽面试中被问的比较频繁的一个问题。今天,我们就一起梳理下有哪些
在日常工作中,面试如果对Spring的必问事务管理功能使用不当,则会造成Spring事务不生效的场景问题。而针对Spring事务不生效的下S效问题,也是事务在跳槽面试中被问的比较频繁的一个问题。
今天,面试我们就一起梳理下有哪些场景会导致Spring事务生效。必问
注:部分内容引用自冰河与猫大人出版的场景《深入理解分布式事务:原理与实战》一书。
文章收录于GitHub和Gitee:
GitHub: https://github.com/sunshinelyz/technology-binghe
Gitee: https://gitee.com/binghe001/technology-binghe
Spring事务不生效总览
简单来说,下S效Spring事务会在几种特定的事务场景下失效,如下图所示。面试
数据库不支持事务
Spring事务生效的必问前提是所连接的数据库要支持事务,如果底层的场景数据库都不支持事务,则Spring的下S效事务肯定会失效。例如,事务如果使用的数据库为MySQL,并且选用了MyISAM存储引擎,则Spring的事务就会失效。
事务方法未被Spring管理
如果事务方法所在的免费信息发布网类没有加载到Spring IOC容器中,也就是说,事务方法所在的类没有被Spring管理,则Spring事务会失效,示例如下。
public class ProductService { @Autowired private ProductDao productDao; @Transactional(propagation = Propagation.REQUIRES_NEW) public void updateProductStockCountById(Integer stockCount, Long id){ productDao.updateProductStockCountById(stockCount, id); } }ProductService类上没有标注@Service注解,Product的实例没有加载到Spring IOC容器中,就会造成updateProductStockCountById()方法的事务在Spring中失效。
方法没有被public修饰
如果事务所在的方法没有被public修饰,此时Spring的事务会失效,例如,如下代码所示。
@Service public class ProductService { @Autowired private ProductDao productDao; @Transactional(propagation = Propagation.REQUIRES_NEW) private void updateProductStockCountById(Integer stockCount, Long id){ productDao.updateProductStockCountById(stockCount, id); } }虽然ProductService上标注了@Service注解,同时updateProductStockCountById()方法上标注了@Transactional(propagation = Propagation.REQUIRES_NEW)注解。
但是,由于updateProductStockCountById()方法为内部的私有方法(使用private修饰),那么此时updateProductStockCountById()方法的事务在Spring中会失效。
同一类中方法调用
如果同一个类中的两个方法分别为A和B,方法A上没有添加事务注解,方法B上添加了 @Transactional事务注解,方法A调用方法B,则方法B的事务会失效。亿华云计算例如,如下代码所示。
@Service public class OrderService { @Autowired private OrderDao orderDao; @Autowired private ProductDao productDao; public void submitOrder(){ //生成订单 Order order = new Order(); long number = Math.abs(new Random().nextInt(500)); order.setId(number); order.setOrderNo("order_" + number); orderDao.saveOrder(order); //减库存 this.updateProductStockCountById(1, 1L); } @Transactional(propagation = Propagation.REQUIRES_NEW) public void updateProductStockCountById(Integer stockCount, Long id){ productDao.updateProductStockCountById(stockCount, id); } }submitOrder()方法和updateProductStockCountById()方法都在OrderService类中,submitOrder()方法上没有标注事务注解,updateProductStockCountById()方法上标注了事务注解,submitOrder()方法调用了updateProductStockCountById()方法,此时,updateProductStockCountById()方法的事务在Spring中会失效。
未配置事务管理器
如果在项目中没有配置Spring的事务管理器,即使使用了Spring的事务管理功能,Spring的事务也不会生效。
例如,没有在项目的配置类中配置如下代码。
@Bean public PlatformTransactionManager transactionManager(DataSource dataSource) { return new DataSourceTransactionManager(dataSource); }此时,Spring的事务就会失效。
方法的事务传播类型不支持事务
如果内部方法的事务传播类型为不支持事务的传播类型,则内部方法的事务在Spring中会失效。
例如,如下代码所示。
@Service public class OrderService { @Autowired private OrderDao orderDao; @Autowired private ProductDao productDao; @Transactional(propagation = Propagation.REQUIRED) public void submitOrder(){ //生成订单 Order order = new Order(); long number = Math.abs(new Random().nextInt(500)); order.setId(number); order.setOrderNo("order_" + number); orderDao.saveOrder(order); //减库存 this.updateProductStockCountById(1, 1L); } @Transactional(propagation = Propagation.NOT_SUPPORTED) public void updateProductStockCountById(Integer stockCount, Long id){ productDao.updateProductStockCountById(stockCount, id); } }由于updateProductStockCountById()方法的事务传播类型为NOT_SUPPORTED,不支持事务,则updateProductStockCountById()方法的云服务器提供商事务会在Spring中失效。
不正确的捕获异常
不正确的捕获异常也会导致Spring的事务失效,示例如下。
@Service public class OrderService { @Autowired private OrderDao orderDao; @Autowired private ProductDao productDao; @Transactional(propagation = Propagation.REQUIRED) public void submitOrder(){ //生成订单 Order order = new Order(); long number = Math.abs(new Random().nextInt(500)); order.setId(number); order.setOrderNo("order_" + number); orderDao.saveOrder(order); //减库存 this.updateProductStockCountById(1, 1L); } @Transactional(propagation = Propagation.REQUIRED) public void updateProductStockCountById(Integer stockCount, Long id){ try{ productDao.updateProductStockCountById(stockCount, id); int i = 1 / 0; }catch(Exception e){ logger.error("扣减库存异常:", e.getMesaage()); } } }updateProductStockCountById()方法中使用try-catch代码块捕获了异常,即使updateProductStockCountById()方法内部会抛出异常,但也会被catch代码块捕获到,此时updateProductStockCountById()方法的事务会提交而不会回滚,并且submitOrder()方法的事务会提交而不会回滚,这就造成了Spring事务的回滚失效问题。
错误的标注异常类型
如果在@Transactional注解中标注了错误的异常类型,则Spring事务的回滚会失效,示例如下。
@Transactional(propagation = Propagation.REQUIRED) public void updateProductStockCountById(Integer stockCount, Long id){ try{ productDao.updateProductStockCountById(stockCount, id); }catch(Exception e){ logger.error("扣减库存异常:", e.getMesaage()); throw new Exception("扣减库存异常"); } }在updateProductStockCountById()方法中捕获了异常,并且在异常中抛出了Exception类型的异常,此时,updateProductStockCountById()方法事务的回滚会失效。
为何会失效呢?这是因为Spring中对于默认回滚的事务异常类型为RuntimeException,上述代码抛出的是Exception异常。
默认情况下,Spring事务中无法捕获到Exception异常,所以此时updateProductStockCountById()方法事务的回滚会失效。
此时可以手动指定updateProductStockCountById()方法标注的事务异常类型,如下所示。
@Transactional(propagation = Propagation.REQUIRED,rollbackFor = Exception.class)这里,需要注意的是:Spring事务注解@Transactional中的rollbackFor属性可以指定 Throwable 异常类及其子类。
本文转载自微信公众号「冰河技术」,可以通过以下二维码关注。转载本文请联系冰河技术公众号。
很赞哦!(77642)
热门文章
站长推荐
尽量不要在域名中出现特殊字符,这样的域名很容易导致访问者输入错误,同时给人留下不专业的印象,降低网站的可信度,并流失大量潜在客户。
详解容器部署ELK7.10,适用于生产
JavaScript的两大类内建数据类型
遇到代码缺陷不要慌,马上教你快速检测和修复
互联网其实拼的也是人脉,域名投资也是一个时效性很强的东西,一个不起眼的消息就会引起整个域名投资市场的动荡,因此拓宽自己的人脉圈,完善自己的信息获取渠道,让自己能够掌握更为多样化的信息,这样才更有助于自己的域名投资。
2020年最佳编程语言Python,第四次获得年度最佳
为什么 Promis 比setTimeout()更快?
为什么Java仍很滋润的活着?