精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

Spring事務失效常見的五種方式及其解決方案

開發 架構
本文總結了Spring 聲明式事務的源碼實現、五種常見的事務失效情況,并提供了相應的解決方案。

一、前言

在Web 開發中,Spring 框架已經成為了眾多開發者的首選。Spring 的聲明式事務管理是其中最重要的特性之一,它可以幫助我們簡化業務邏輯的復雜度,并且確保在出現異常情況時數據的一致性。

事務失效情況很常見,但我們只要注意,就可以避免事情發生!在本文中,我將詳細地介紹 Spring 聲明式事務的源碼實現和事務失效常見的五種情況,并給出有效的解決方案。

其實我們常說的事務失效是聲明式事務(@Transactional)的失效,本文也是從聲明式事務來進行演示的!

通過本文的學習,你將掌握如何正確地使用 Spring 的事務管理,減少生產事故。

「一定要保持數據一致性」。

二、@Transactional注解參數解讀

我們拿出幾個經常使用的參數來簡單介紹一下:

  • propagation:指定事務的傳播行為。其取值包括 REQUIRED、SUPPORTS、MANDATORY、REQUIRES_NEW、NOT_SUPPORTED、NEVER 和 NESTED 等。默認為 REQUIRED。 其中,REQUIRED 表示如果當前已經存在一個事務,則加入該事務,否則新建一個事務;而 REQUIRES_NEW 表示新建一個獨立的事務,如果當前已經存在事務,則掛起當前事務。后面就不一一說了,大家可以自行百度哈!
  • isolation:指定事務的隔離級別。其取值包括 DEFAULT、READ_UNCOMMITTED、READ_COMMITTED、REPEATABLE_READ 和 SERIALIZABLE 等。默認為 DEFAULT。 其中,DEFAULT 表示采用數據庫的默認隔離級別.
  • timeout:指定事務的超時時間,單位為秒。默認為 -1,表示不設置超時時間。如果在規定時間內事務還未完成,則拋出 TransactionTimedOutException 異常。
  • readOnly:指定事務是否只讀,即是否允許修改數據。默認為 false,表示可以進行數據修改操作。如果將其設置為 true,則表示該事務僅能進行數據查詢操作,不能進行數據修改操作,這樣可以提高并發性能。
  • rollbackFor:指定哪些異常需要回滾事務。其取值為一個 Class 數組,其中每個元素表示一個異常類型。默認為空,表示只有拋出 RuntimeException 或 Error 類型的異常時才回滾事務。
  • noRollbackFor:指定哪些異常不需要回滾事務。其取值為一個 Class 數組,其中每個元素表示一個異常類型。默認為空,表示拋出任何異常都回滾事務。

三、聲明式事務源碼實現

聲明式事務實現類為:TransactionInterceptor ,下面我們來一起看看這個類!

源碼版本為Springboot2.7.1。

public class TransactionInterceptor extends TransactionAspectSupport 
 implements MethodInterceptor, Serializable{}

TransactionInterceptor UML圖:

圖片

聲明式事務主要是通過AOP實現,主要包括以下幾個節點:

  1. 啟動時掃描@Transactional注解:在啟動時,Spring Boot會掃描所有使用了@Transactional注解的方法,并將其封裝成TransactionAnnotationParser對象。
  2. AOP 來實現事務管理的核心類依然是 TransactionInterceptor。TransactionInterceptor 是一個攔截器,用于攔截使用了 @Transactional 注解的方法
  3. 將TransactionInterceptor織入到目標方法中:在AOP編程中,使用AspectJ編寫切面類,通過@Around注解將TransactionInterceptor織入到目標方法中。
  4. 在目標方法執行前創建事務:在目標方法執行前,TransactionInterceptor會調用PlatformTransactionManager創建一個新的事務,并將其納入到當前線程的事務上下文中。
  5. 執行目標方法:在目標方法執行時,如果發生異常,則將事務狀態標記為ROLLBACK_ONLY;否則,將事務狀態標記為COMMIT。
  6. 提交或回滾事務:在目標方法執行完成后,TransactionInterceptor會根據事務狀態(COMMIT或ROLLBACK_ONLY)來決定是否提交或回滾事務。

源碼:

@Override
@Nullable
public Object invoke(MethodInvocation invocation) throws Throwable {
 // Work out the target class: may be {@code null}.
 // The TransactionAttributeSource should be passed the target class
 // as well as the method, which may be from an interface.
 Class<?> targetClass = (invocation.getThis() != null ? AopUtils.getTargetClass(invocation.getThis()) : null);

 // Adapt to TransactionAspectSupport's invokeWithinTransaction...
 return invokeWithinTransaction(invocation.getMethod(), targetClass, new CoroutinesInvocationCallback() {
  @Override
  @Nullable
  public Object proceedWithInvocation() throws Throwable {
   return invocation.proceed();
  }
  @Override
  public Object getTarget() {
   return invocation.getThis();
  }
  @Override
  public Object[] getArguments() {
   return invocation.getArguments();
  }
 });
}

下面是核心處理方法,把不太重要的代碼忽略了,留下每一步的節點。

@Nullable
protected Object invokeWithinTransaction(Method method, @Nullable Class<?> targetClass,
 final InvocationCallback invocation) throws Throwable {
 // 獲取事務屬性
 final TransactionManager tm = determineTransactionManager(txAttr);
 // 準備事務
 TransactionInfo txInfo = prepareTransactionInfo(ptm, txAttr, joinpointIdentification, status);
 // 執行目標方法
 Object retVal = invocation.proceedWithInvocation();
  // 回滾事務
 completeTransactionAfterThrowing(txInfo, ex);
 // 提交事務
 commitTransactionAfterReturning(txInfo);
}

四、五種失效和解決方案

圖片

下面我們從幾個情況來給大家展示失效場景并給出解決方案。

1、類沒有被 Spring 管理

public class UserServiceImpl implements UserService {

    @Autowired
    private UserDao userDao;

    @Override
    @Transactional(rollbackFor = Exception.class)
    public void addUser(User user) {
        userDao.addUser(user);
    }
}

如上代碼所示,UserServiceImpl 類沒有被聲明為 Spring Bean,因此其中的 addUser() 方法無法受到 Spring 事務管理的保護。 我們使用Spring,要把類交給Spring進行管理,不然是無法生效!

「解決方案:」 交給spring進行管理bean,在類上添加:@Service!

2、方法不是public修飾

@Service
public class UserService {

    @Autowired
    private UserDao userDao;

    @Transactional(rollbackFor = Exception.class)
    protected void addUser(User user) {
        userDao.addUser(user);
    }
}

我們上面說了聲明式事務是基于AOP實現的,AOP是通過代理模式實現的,即為目標對象生成一個代理對象,當調用代理對象的方法時,會自動添加事務的控制代碼。 在這種情況下,如果事務注釋所在的方法不是public的,則無法生成代理對象,因此事務代碼將無法添加到方法執行前后,導致事務失效。

其實這種情況還是不經常這么使用,我們基本都是使用接口和實現大部分都是public修飾的!

「解決方案:」 使用public來修飾方法。

3、異常被捕獲并處理了

@Service
public class UserServiceImpl implements UserService {

    @Autowired
    private UserDao userDao;

    @Override
    @Transactional(rollbackFor = Exception.class)
    public void addUser(User user) {
        try {
            userDao.addUser(user);
        } catch (Exception e) {
            // 處理異常,但沒有拋出或重新拋出異常
            log.error("add user error", e);
        }
    }
}

如上代碼所示,如果 userDao.addUser() 方法拋出異常,但是在 UserServiceImpl.addUser() 中被捕獲并處理了,事務檢測不到有異常拋出,那么事務不會回滾。

「解決方案:」 catch 處理完成后,在重新把異常在拋出去:throw e。

4、同一個類中,方法內部調用

@Service
public class UserServiceImpl implements UserService {

    @Autowired
    private UserDao userDao;

    @Override
    public void addUser(User user) {
        doAddUser(user);
    }

    @Transactional(rollbackFor = Exception.class)
    public void doAddUser(User user) {
        userDao.addUser(user);
    }
}

Spring使用代理來實現事務控制,但是這種方法直接調用了this對象的方法,則無法通過代理來攔截該方法調用,從而使得事務失效。

「解決方案:」

推薦使用有兩種:

  • 使用ApplicationContext來獲取當前bean對象來調用doAddUser方法。
  • 在addUser方法加上@Transactional(rollbackFor = Exception.class)。

網上還有一些使用AopContext.currentProxy()拿到代理對象的、自己注入自己的、抽到單獨的bean里的 這里小編不是很推薦!

方法一完整展示:

如果覺得Service里注入ApplicationContext 不優雅,可以抽到單獨的工具bean里!

@Service
public class UserServiceImpl implements UserService {

 @Autowired
    private UserDao userDao;
    @Autowired
 private ApplicationContext applicationContext;

    @Override
    public void addUser(User user) {
     UserServiceImpl userService = applicationContext.getBean(UserServiceImpl.class);
        userService.doAddUser(user);
    }

    @Transactional(rollbackFor = Exception.class)
    public void doAddUser(User user) {
        userDao.addUser(user);
    }
}

5、MySQL存儲引警不支持事務

MyISAM 存儲引擎是 MySQL 的一種存儲引擎,它是 MySQL 5.1 版本之前的默認存儲引擎,它是不支持事務的。從 MySQL 5.5 版本開始,InnoDB 成為了 MySQL 的默認存儲引擎。我們想使用也可以切換到MyISAM引擎。

「解決方案:」 把mysql換到5.5以上使用InnoDB 存儲引擎。

「補充使用MyISAM 方式:」

  • 表從 InnoDB 引擎轉換為 MyISAM 引擎:使用 ALTER TABLE 命令來更改表的引擎類型。
ALTER TABLE table_name ENGINE = MyISAM;
  • 默認的存儲引擎設置為 MyISAM, 可以在 MySQL 配置文件中設置 default-storage-engine 參數。
default-storage-engine=MyISAM
  • 創建表時指定MyISAM 引擎 要將表的引擎類型設置為 MyISAM,請在 CREATE TABLE 語句中包含 ENGINE = MyISAM 子句
CREATE TABLE table_name (
    column1 datatype,
    column2 datatype,
    ...
) ENGINE = MyISAM;

五、總結

本文總結了Spring 聲明式事務的源碼實現、五種常見的事務失效情況,并提供了相應的解決方案。

當然還有很多情況:被final修飾、多線程調用、傳播行為使用不當、拋的異常不對應等等

理解 Spring 事務機制的,深入了解 Spring 事務的內部原理。同時,在使用聲明式事務的過程中,我們也可以針對自己的業務場景進行定制化的配置,比如指定特定的事務傳播機制、設置超時時間等,這些都有助于更好地應對復雜的業務場景和代碼需求。這樣才能真正地提高系統的可維護性、可擴展性和穩定性。

責任編輯:姜華 來源: 小王博客基地
相關推薦

2011-02-28 13:51:30

Spring事物配置

2009-06-19 18:26:38

Spring事務配置

2011-11-25 10:25:27

SpringJava

2020-03-31 16:13:26

分布式事務方案TCC

2022-12-19 08:23:34

2022-03-30 07:52:16

.NET應用程序C#

2018-10-10 10:23:53

數據庫RedisNoSQL

2023-04-14 14:54:29

2018-05-04 07:36:35

醫療行業物聯網IoT

2022-12-06 10:39:43

Spring事務失效

2020-08-20 20:51:17

打散算法打散法原算法

2024-03-26 12:08:53

分布式事務存儲

2021-06-28 09:00:00

Kubernetes容器云計算

2019-06-14 05:00:05

2022-02-23 12:07:20

分布式Spark數據傾斜

2024-09-09 08:29:25

2010-09-09 10:29:47

2023-05-06 15:32:04

2025-04-29 04:00:00

分布式事務事務消息

2020-06-08 22:33:42

物聯網IOT物聯網實施
點贊
收藏

51CTO技術棧公眾號

人人妻人人藻人人爽欧美一区| 国产日韩第一页| 久久这里只有精品9| 欧美日韩mv| 亚洲免费av网址| 中文字幕成人在线视频| av福利在线导航| 国产精品美女久久久久久| 国产精品免费观看高清| 中文字幕 自拍偷拍| 国产综合网站| 最近2019中文字幕一页二页 | 看黄在线观看| 中文字幕亚洲一区二区av在线| 国产乱码一区| 亚洲专区在线播放| 国产精品综合色区在线观看| 欧美成人午夜激情视频| 高清国产在线观看| 欧美国产不卡| 欧美一区二区在线不卡| 亚洲 欧美 日韩系列| 久热在线观看视频| 亚洲综合av网| 中文字幕在线乱| 91在线看黄| 久久久精品综合| 国产一区二区精品在线| 超碰免费在线97| 久久狠狠亚洲综合| 国产精品偷伦视频免费观看国产 | 久久久久网址| 黄色av一区二区三区| 国产一区二区不卡| 成人网欧美在线视频| 中文在线a天堂| 日韩制服丝袜av| 日本一区二区三区四区视频| 国产精品成人网站| 一区免费在线| 久久久久国产精品免费网站| 久久国产精品二区| 欧美日本不卡高清| 欧美激情一二三| 久久久精品99| 精品91久久久久| 97色在线视频观看| 日本熟妇毛耸耸xxxxxx| 亚洲国产国产亚洲一二三| 久久久久久国产三级电影| 久久精品国产av一区二区三区| 欧美黄色精品| 欧美激情在线视频二区| 国产精久久久久久| 亚洲全部视频| 情事1991在线| 中文在线免费看视频| 免费成人av在线| 国产乱人伦真实精品视频| 中文字幕久久网| 久久av资源站| 97操在线视频| 五月天婷婷激情网| 久久久久国产成人精品亚洲午夜| 亚洲 国产 欧美一区| 免费看美女视频在线网站| 中文字幕精品综合| 潘金莲一级淫片aaaaaa播放1| av网站大全在线| 亚洲观看高清完整版在线观看| 宅男噜噜噜66一区二区| 99久久免费视频.com| 国产在线播放一区二区| 香蕉视频成人在线| 国产视频在线观看一区二区三区 | 欧美亚洲在线日韩| 亚洲欧美在线免费观看| 天堂网中文在线观看| 欧美一区二区| 97超碰蝌蚪网人人做人人爽 | 日韩av网站免费在线| 国产日韩在线播放| 国模私拍视频在线| 国产区在线观看成人精品| 国产又大又长又粗又黄| 538视频在线| 欧美性欧美巨大黑白大战| 性鲍视频在线观看| 少妇精品导航| 久久亚洲精品成人| 国产三级精品三级在线观看| 精品一二三四区| 久久99精品久久久久久青青日本| 夜级特黄日本大片_在线| 一二三四区精品视频| 欧美激情国产精品日韩| 日韩欧洲国产| 影音先锋欧美精品| 日产欧产va高清| 毛片av一区二区三区| 精品免费视频123区| 蜜桃视频在线观看免费视频网站www | 日本一区二区免费看| 精灵使的剑舞无删减版在线观看| 色爱区综合激月婷婷| 亚洲国产精品狼友在线观看| 日韩欧美在线中字| 秋霞av国产精品一区| 亚洲精品久久久久久无码色欲四季| 国产日本一区二区| 欧美成人一区二区在线观看| 精品视频在线观看网站| 中日韩美女免费视频网址在线观看| 国产亚洲成人精品| 国产一区二区三区日韩 | 婷婷五月在线视频| 精品国产鲁一鲁一区二区张丽| 在线视频观看一区二区| 国产成人3p视频免费观看| 欧美精品久久久久久久| 国产精选久久久| 国产精品三级电影| 婷婷丁香激情网| 免费av一区二区三区四区| 午夜精品理论片| 亚洲黄色在线观看视频| 亚洲精品国久久99热| 男生操女生视频在线观看| 日本精品三区| 国产精品激情自拍| 国产人成在线观看| 一本久道久久综合中文字幕| 成人在线电影网站| 亚洲激情影院| 国产一区精品在线| 成年人黄色大片在线| 欧美精品一区男女天堂| 久久香蕉精品视频| 粉嫩久久99精品久久久久久夜| 日本久久高清视频| 欧美激情三级| 欧美疯狂做受xxxx高潮| www.色视频| 亚洲国产日产av| xxxxxx黄色| 国产欧美日韩一级| 欧美成人一区二区在线| 日韩性xxx| 在线视频一区二区| 亚洲av综合一区| 亚洲素人一区二区| 中文字幕久久久久久久| 好吊一区二区三区| 国产女人水真多18毛片18精品 | 中文字幕免费视频| 日韩精品免费视频人成| 亚洲乱码一区二区三区| 国产精品2区| 欧美激情免费看| 日韩欧美在线观看一区二区| 91久久香蕉国产日韩欧美9色| 极品人妻videosss人妻| 国产专区欧美精品| 99er在线视频| 亚洲婷婷影院| 国产日韩欧美视频| 久久大胆人体| 亚洲欧美日韩中文视频| 在线亚洲欧美日韩| 夜夜揉揉日日人人青青一国产精品| 中文字幕在线视频播放| 久久久久国产精品一区三寸| 一区二区三视频| 这里视频有精品| 国产99久久精品一区二区永久免费 | 激情在线视频播放| 亚洲欧美激情视频| 国产精品久久久久久在线| 一个色妞综合视频在线观看| 伊人网在线视频观看| 国精产品一区一区三区mba桃花| 欧美久久在线观看| 成人女性视频| 国产区一区二区三区| 亚洲精品一区三区三区在线观看| 久久亚洲精品网站| 黄色在线播放| 日韩精品一区国产麻豆| 69视频免费看| 亚洲综合色区另类av| av网在线播放| 成人深夜福利app| 制服丝袜综合网| 夜夜嗨一区二区| 免费观看黄色大片| 九九综合久久| 国产精品国产精品国产专区蜜臀ah | 久久精品亚洲欧美日韩精品中文字幕| 国产在线一区二区三区四区| 四虎国产精品免费久久5151| 日韩免费在线观看视频| 黄网站在线观| 久久在线免费观看视频| 国产精品一区二区婷婷| 亚洲福利小视频| 97人妻精品一区二区三区动漫| 欧美午夜性色大片在线观看| 破处女黄色一级片| 中文字幕欧美激情一区| 大地资源二中文在线影视观看| 国产一区二区视频在线播放| 日本熟妇人妻中出| 国产亚洲精品v| 黄色成人在线免费观看| 91精品久久久久久久久久不卡| 日本不卡免费新一二三区| 久久这里只有精品一区二区| 91福利视频导航| 日本一区二区中文字幕| 国产精品成人一区二区三区吃奶| 国产污视频在线播放| 欧美日韩国产91| 91精品国产91久久久久久青草| 日韩在线观看免费网站| 成年人在线免费观看| 亚洲欧美日本另类| 青青久在线视频免费观看| 亚洲精品国产电影| 天堂中文在线看| 亚洲成人精品久久| 亚洲精品一区二区口爆| 日韩欧美在线影院| 国产黄色美女视频| 日韩精品一区二区在线| 成人激情四射网| 欧美成人精品福利| 亚洲成熟女性毛茸茸| 日韩欧美成人午夜| 亚洲国产精品国自产拍久久| 日韩一级高清毛片| 超碰在线播放97| 亚洲白拍色综合图区| 国产91免费在线观看| 亚洲国产精品字幕| 日韩电影在线观看完整版| 精品一区二区三区四区| 九色国产在线观看| 最近更新的2019中文字幕| 欧美激情黑人| 欧美成人高清视频| bbw在线视频| 国产91精品久| 天然素人一区二区视频| 国产精品自在线| 成人精品视频在线观看| 99porn视频在线| 美女一区二区在线观看| 久久影视中文粉嫩av| 欧美男同视频网| 亚洲欧洲一区二区在线观看| 91精品啪在线观看国产18| 国产一级大片免费看| 亚洲黄色毛片| 精品免费国产一区二区| 久久99久久99精品免视看婷婷| 国内精品国产三级国产aⅴ久| 成人免费视频国产在线观看| 精品人妻一区二区三区视频| 国产精品美女久久久久av爽李琼| 欧美大片xxxx| 午夜精品福利久久久| 波多野结衣不卡| 日韩一区二区麻豆国产| 五月婷婷六月丁香综合| 中文字幕欧美国内| 欧美亚洲天堂| 国产999精品视频| 亚洲精品aⅴ| 日韩av一区二区三区在线| 亚洲精品tv久久久久久久久久| 3d动漫一区二区三区| 老色鬼精品视频在线观看播放| 日批视频免费看| 国产欧美精品在线观看| 免费一级a毛片夜夜看| 色猫猫国产区一区二在线视频| 国产理论视频在线观看| 国产视频精品免费播放| 黄色片网站在线| 欧美在线激情网| 国产精品一区二区三区av| 免费在线观看91| 欧美韩国一区| 九色porny自拍| 91丨porny丨蝌蚪视频| xxxx日本少妇| 欧美午夜不卡视频| 午夜影院在线视频| 欧美日本黄视频| 青青久久精品| 欧美日韩在线观看一区| 欧美1区免费| 亚洲色图 在线视频| 91丨porny丨国产入口| 久草视频在线资源站| 欧美乱妇15p| 国产精品影院在线| 欧美中文字幕在线视频| 99ri日韩精品视频| 午夜啪啪免费视频| 日韩不卡手机在线v区| 影音先锋黄色资源| 亚洲综合999| 国产视频手机在线| 日韩在线国产精品| 国产91亚洲精品久久久| 久久免费视频1| 国产欧美一级| 性色av蜜臀av浪潮av老女人| 亚洲精品福利视频网站| 国产精品免费无遮挡| 在线观看欧美www| 电影一区二区三| 久久精品国产美女| av成人天堂| 国产麻豆xxxvideo实拍| 性欧美疯狂xxxxbbbb| 黄色美女一级片| 欧美精品成人91久久久久久久| 精品91福利视频| 艳母动漫在线观看| 国产资源精品在线观看| tube国产麻豆| 欧美一区二区日韩| 午夜伦理在线视频| 亚洲在线一区二区| 欧美日韩午夜| av天堂一区二区| 污片在线观看一区二区| 天天色天天操天天射| 57pao国产精品一区| 伊人久久大香线蕉综合网蜜芽| 久久精品99国产| 欧美国产国产综合| 91久久精品无码一区二区| 日韩中文av在线| 麻豆一二三区精品蜜桃| 永久免费网站视频在线观看| 国产成人综合在线播放| 欧美日韩精品亚洲精品| 精品国产sm最大网站| 国产高潮在线| 色狠狠久久av五月综合|| 精品亚洲aⅴ乱码一区二区三区| 99鲁鲁精品一区二区三区| 日韩欧美国产高清| 亚洲一二三四| 中文字幕一区综合| 成人黄页在线观看| 国产精品视频一区在线观看| 中日韩美女免费视频网站在线观看 | 亚洲人成人无码网www国产| 欧美亚洲综合另类| www在线观看播放免费视频日本| 成人综合av网| 久久激情婷婷| 国产天堂av在线| 亚洲国产高清福利视频| av高清一区| 精品无码av无码免费专区| 97久久超碰国产精品| 波多野结衣家庭主妇| 欧美成人精品三级在线观看| 欧美成人专区| 亚洲天堂网2018| 午夜电影一区二区| a黄色在线观看| 高清不卡日本v二区在线| 日日夜夜免费精品视频| 天天看片中文字幕| 亚洲免费伊人电影在线观看av| 日韩美香港a一级毛片| 美脚丝袜脚交一区二区| 国产精品免费丝袜| 人妻va精品va欧美va| 国产精品一区二区女厕厕| 一本综合久久| 日韩精品一区二区亚洲av性色 | 素人av在线| 国产一区自拍视频| 国产在线精品一区二区不卡了 | 日本欧美一区二区| 久久久久久久久久久久久久久久久| 亚洲精品一区中文字幕乱码| 精品国产一区二| 嫩草av久久伊人妇女超级a| 亚洲r级在线视频| av片在线观看网站| 色99中文字幕| 91麻豆国产香蕉久久精品|