成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

面試必問|哪些場景下Spring的事務會失效?

開發 架構
在日常工作中,如果對Spring的事務管理功能使用不當,則會造成Spring事務不生效的問題。而針對Spring事務不生效的問題,也是在跳槽面試中被問的比較頻繁的一個問題。

[[440018]]

在日常工作中,如果對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 異常類及其子類。

本文轉載自微信公眾號「冰河技術」,可以通過以下二維碼關注。轉載本文請聯系冰河技術公眾號。

 

責任編輯:武曉燕 來源: 冰河技術
相關推薦

2024-01-05 14:20:55

MySQL索引優化器

2023-06-30 07:58:07

Spring數據源事務

2021-04-15 08:01:27

Spring聲明式事務

2022-02-14 16:53:57

Spring項目數據庫

2023-02-02 07:06:10

2022-04-13 20:53:15

Spring事務管理

2023-07-05 08:45:18

Spring事務失效場景

2022-09-22 09:57:20

Spring事務失效

2024-09-09 08:29:25

2024-01-29 08:28:01

Spring事務失效

2022-09-20 22:27:08

事務失效public 修飾

2023-02-06 07:01:51

2021-09-04 07:56:44

Spring事務失效

2022-09-14 19:50:22

事務場景流程

2021-12-27 08:22:18

Kafka消費模型

2025-02-10 00:27:54

2023-06-07 08:08:43

JVM內存模型

2020-07-28 08:59:22

JavahreadLocal面試

2021-12-09 12:22:28

MyBatis流程面試

2023-08-15 15:33:29

線程池線程數
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 亚洲国产成人精品在线 | 在线看黄免费 | 中文字幕国产精品 | 欧美午夜精品 | 国产精品成人一区 | 中日韩毛片 | 亚洲一区国产精品 | 日韩午夜一区二区三区 | 欧美a在线看 | 婷婷五月色综合 | 久久国产日本 | 国产成人在线免费 | 欧美极品在线 | 欧美日韩国产精品激情在线播放 | 蜜桃av一区二区三区 | 国产日韩久久久久69影院 | 麻豆久久久久 | 亚洲欧美综合精品久久成人 | 久久久精品一区二区 | 久久成人免费 | 国产精品美女久久久免费 | 一级片在线免费播放 | 亚洲国产成人精品女人久久久 | hitomi一区二区三区精品 | 国产精品欧美一区二区 | 在线天堂免费中文字幕视频 | 一级片在线观看 | 亚洲专区在线 | 亚洲网一区| 天天影视网天天综合色在线播放 | 99久久国产| 色婷婷久久综合 | 亚洲欧美在线观看 | 日韩中文字幕一区二区 | 国产电影一区二区 | 亚洲精品美女视频 | 日本一道本视频 | 日韩视频区| 中文字幕一区二区三区四区 | 九九热在线观看视频 | 中国一级大黄大片 |