Java @Transactional與synchronized使用的問題
引言
@Transactional是spring通過aop讓我們輕松實現(xiàn)事務控制的一個注解;而synchronized是實現(xiàn)同步的java關鍵字;但是它們兩個不能一起使用,一起使用會出現(xiàn)
synchronized失效的問題,這里簡單記錄一下這個問題;
發(fā)現(xiàn)問題
我在impl中實現(xiàn)一個功能邏輯時,為了保證冪等性,在方法中使用synchronized保證同一個用戶短時間內多次請求只能串行執(zhí)行,因為數(shù)據(jù)變動涉及多張表,所以我又加了一個@Transactional注解,偽代碼如下:
@Transactional public void demo() { ...一些操作... synchronized(lock) { ...數(shù)據(jù)庫讀寫操作... } }
然后測試時就發(fā)現(xiàn)synchronized沒有生效,確認代碼邏輯沒有問題后,查詢了資料發(fā)現(xiàn)了問題;
問題原因
這里不過多解釋@Transactional注解底層,感興趣可以自行查閱資料;
主要原因就是@Transactional注解通過aop實現(xiàn)事務管理,當標注該注解的方法執(zhí)行完成后才提交事務,而synchronized代碼塊又是在一個事務內,就會出現(xiàn)第一個線程釋放鎖后但是事務還沒提交,第二個線程就進入同步代碼塊獲取到未提交的數(shù)據(jù)庫數(shù)據(jù);
大致如圖:
解決問題
大致思路
解決方法很簡單,既然問題出在事務未提交,那么只要把對應事務操作的代碼單獨抽取出來,封裝成一個單獨的方法,在synchronized中調用該方法即可;
如圖:
偽代碼為:
public void demo() { ...一些操作... synchronized(lock) { databaseOption(); // 調用數(shù)據(jù)庫操作方法 } } @Transactional public void databaseOption() { ...數(shù)據(jù)庫讀寫操作... }
注意:@Transactional注解修飾的方法需要是public權限;
雖然這樣寫好像解決了事務未提交的問題,但是這樣寫會存在新的問題;上面這種是這兩種方法都寫在serviceImpl中,但是這樣調用databaseOption方法就會出現(xiàn)@Transactional事務不生效的情況;
@Transactional事務不生效問題
所以在同一個類內部調用@Transactional標注的方法事務也不會開啟,原因是:
@Transactional事務管理是基于動態(tài)代理對象的代理邏輯實現(xiàn)的,那么如果在類內部調用類內部的事務方法,這個調用事務方法的過程并不是通過代理對象來調用的,而是直接通過this對象來調用方法,繞過的代理對象,肯定就是沒有代理邏輯了
依然是@Transactional的底層原理,可以好好研究一下這個注解,面試就有的聊了;
那么解決方法就是:
不要把由@Transactional修飾的databaseOption方法和調用它的方法放到同一個類中;這里你可以多寫個service放databaseOption方法,但是這樣好像沒有什么意義;我選擇的是把同步代碼塊放到controller中,在controller中調用serviceImpl中的databaseOption方法;
偽代碼:
controller類
@RestController public class TestController { @Resource private TestService testService; @PostMapping("/test") public String testInterface() { ...一些操作... synchronized(lock) { testService.databaseOption(); // 調用數(shù)據(jù)庫操作方法 } } }
service類
@Service("testService") public class TestServiceImpl implements TestService { @Transactional public void databaseOption() { ...數(shù)據(jù)庫讀寫操作... } }
這樣就能保證@Transactional事務生效;但是這樣寫的缺點就是一些邏輯會被拆分到controller中,可讀性會稍差點;
總結
這就是這兩天我踩的坑,總的來說所有問題的出現(xiàn)都是由于對@Transactional這個注解理解的不透徹,以后還是要了解該注解的實現(xiàn)原理;
到此這篇關于Java @Transactional與synchronized使用的問題的文章就介紹到這了,更多相關Java @Transactional與synchronized內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
- Java同步鎖Synchronized底層源碼和原理剖析(推薦)
- java同步鎖的正確使用方法(必看篇)
- 95%的Java程序員人都用不好Synchronized詳解
- Java?synchronized同步關鍵字工作原理
- Java synchronized偏向鎖的概念與使用
- Java?synchronized輕量級鎖實現(xiàn)過程淺析
- Java synchronized重量級鎖實現(xiàn)過程淺析
- Java?synchronized與死鎖深入探究
- Java synchronized與CAS使用方式詳解
- 淺析Java關鍵詞synchronized的使用
- synchronized及JUC顯式locks?使用原理解析
- java鎖synchronized面試常問總結
- Java?HashTable與Collections.synchronizedMap源碼深入解析
- Java?Synchronized鎖的使用詳解
- AQS加鎖機制Synchronized相似點詳解
- Java必會的Synchronized底層原理剖析
- 一個例子帶你看懂Java中synchronized關鍵字到底怎么用
- 詳解Java?Synchronized的實現(xiàn)原理
- Synchronized?和?ReentrantLock?的實現(xiàn)原理及區(qū)別
- Java同步鎖synchronized用法的最全總結
相關文章
解決mybatis使用char類型字段查詢oracle數(shù)據(jù)庫時結果返回null問題
這篇文章主要介紹了mybatis使用char類型字段查詢oracle數(shù)據(jù)庫時結果返回null問題的解決方法,本文給大家介紹的非常詳細,具有一定的參考借鑒價值,需要的朋友可以參考下2018-06-06Spring中bean的初始化和銷毀幾種實現(xiàn)方式詳解
這篇文章主要介紹了Spring中bean的初始化和銷毀幾種實現(xiàn)方式詳解,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2020-11-11SpringBoot集成I18n國際化文件在jar包外生效問題
這篇文章主要介紹了SpringBoot集成I18n國際化文件在jar包外生效問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-04-04UniApp?+?SpringBoot?實現(xiàn)微信支付和退款功能
這篇文章主要介紹了UniApp?+?SpringBoot?實現(xiàn)微信支付和退款功能,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2022-06-06Spring Boot集成SpringFox 3.0與Pageable參數(shù)處理方法
這篇文章主要介紹了Spring Boot集成SpringFox 3.0與Pageable參數(shù)處理,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2023-10-10springboot?+mybatis?使用PageHelper實現(xiàn)分頁并帶條件模糊查詢功能
這篇文章主要介紹了springboot?+mybatis?使用PageHelper實現(xiàn)分頁并帶條件模糊查詢功能,本文通過實例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2022-02-02