Spring事务总结-Spring的事务失效场景-《Java笔记》

admin 2025-10-19 04:18:50 编程 来源:ZONE.CI 全球网 0 阅读模式

Java Spring 事务失效

Spring事务不生效总览

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

数据库不支持事务

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

事务方法未被Spring管理

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

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

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

方法没有被public修饰

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

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

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);
  4. }

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

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

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

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

由于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. @Transactional(propagation = Propagation.REQUIRED)
  8. public void submitOrder(){
  9. //生成订单
  10. Order order = new Order();
  11. long number = Math.abs(new Random().nextInt(500));
  12. order.setId(number);
  13. order.setOrderNo("order_" + number);
  14. orderDao.saveOrder(order);
  15. //减库存
  16. this.updateProductStockCountById(1, 1L);
  17. }
  18. @Transactional(propagation = Propagation.REQUIRED)
  19. public void updateProductStockCountById(Integer stockCount, Long id){
  20. try{
  21. productDao.updateProductStockCountById(stockCount, id);
  22. int i = 1 / 0;
  23. }catch(Exception e){
  24. logger.error("扣减库存异常:", e.getMesaage());
  25. }
  26. }
  27. }

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. }
  9. }

updateProductStockCountById()方法中捕获了异常,并且在异常中抛出了Exception类型的异常,此时,updateProductStockCountById()方法事务的回滚会失效。为何会失效呢?这是因为Spring中对于默认回滚的事务异常类型为RuntimeException,上述代码抛出的是Exception异常。默认情况下,Spring事务中无法捕获到Exception异常,所以此时updateProductStockCountById()方法事务的回滚会失效。此时可以手动指定updateProductStockCountById()方法标注的事务异常类型,如下所示。

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

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

以太坊cppgolang区别 编程

以太坊cppgolang区别

以太坊是一种去中心化的开源平台,它采用智能合约技术,旨在构建和运行不受干扰的分布式应用程序。作为目前最受欢迎的区块链平台之一,以太坊提供了多种编程语言的支持,其
progolang 编程

progolang

Go语言(Golang)是由Google开发的一门静态类型编程语言。作为一名专业的Golang开发者,我深知这门语言的优势和特点。在本文中,我将介绍Golang
golangn个发送者 编程

golangn个发送者

Golang是一种开源的编程语言,由Google团队开发,旨在提高程序的并发性和简化软件开发过程。在Go语言中,有时需要向多个接收者发送信息。本文将介绍如何在G
golang技能图谱 编程

golang技能图谱

从互联网行业的快速发展到人工智能技术的日益成熟,各种编程语言也应运而生。而在这众多的编程语言中,Golang(即Go)作为一门强大且高效的开发语言备受关注。Go
评论:0   参与:  0