SpringBoot異步與事務一起使用的問題解決
最近遇到的一個場景,在一個被 @Transactional 注解的方法A中中調用了一個被 @Async 注解標記的方法B,由于方法B 在執(zhí)行時方法A 的事務沒有提交,但是方法B在執(zhí)行過程中獲取不到方法A中尚未提交的數據,從而最終倒是方法B執(zhí)行異常。
@Transactional public void create(User user){ ? // 如果用戶已存在,則先刪除 ? delete(user.id); ? // 創(chuàng)建用戶 ? int userId = insert(user); ? // ?更新用戶信息 ? update(userId); } @Async public void update(Integer userId){ ? Icon icon = getUserIcon(userId); ? // 更新用戶圖片 ? updateUserPohot(userId,icon); }
像上面的代碼,我為創(chuàng)建用戶的方法上標記了@Transactional事務注解,然后在其中調用了update()更新方法,這個方法上標記了@Async 注解。這樣代碼雖然看起來沒有什么問題,但是實際在執(zhí)行update()方法時,由于是其他線程去執(zhí)行的,就會導致有可能 create()方法對應的事務還沒有提交,update() 方法就無法讀取到新插入的 user 記錄,從而導致更新失敗。
解決方案
通過調整邏輯保證事務在調用異步方法前被提交
這個問題的原因是由于 @Transactional 和 @Async 注解一起使用導致的,那么我們可以從這個方向入手,首先我們可以先確認將create()方法的事務提交后,然后再去執(zhí)行異步更新方法:
public void create(User user){ ? int userId = doCreate(user); ? // ?更新用戶信息 ? update(userId); } @Transactional public void doCreate(User user){ ? ? // 如果用戶已存在,則先刪除 ? delete(user.id); ? // 創(chuàng)建用戶 ? return insert(user); } @Async public void update(Integer userId){ ? Icon icon = getUserIcon(userId); ? // 更新用戶圖片 ? updateUserPohot(userId,icon); }
異步方法放在事務方法外調用,這樣異步方法就能夠讀取到已經提交的事務數據了。
或者我們還可以使用TransactionTemplate來代替 @Transactional 注解:
@Autowired TransactionTemplate transactionTemplate; public void create(User user){ ? int userId = transactionTemplate.execute(status->{ ? ? // 如果用戶已存在,則先刪除 ? ? delete(user.id); ? ? // 創(chuàng)建用戶 ? ? return insert(user); ? }); ? // ?更新用戶信息 ? update(userId); } @Async public void update(Integer userId){ ? Icon icon = getUserIcon(userId); ? // 更新用戶圖片 ? updateUserPohot(userId,icon); }
通過 TransactionTemplate細化了事務粒度,可以保證在調用異步方法前事務已經被提交。
上面的方案基本都能 解決問題,下面是從網上找到的,spring 給出的解決方案:
@Transactional public void create(User user){ ? // 如果用戶已存在,則先刪除 ? delete(user.id); ? // 創(chuàng)建用戶 ? int userId = insert(user); ? TransactionSynchronizationManager.registerSynchronization(new TransactionSynchronization() { ? ? @Override ? ? public void afterCommit() { ? ? ? // ?更新用戶信息 ? ? ? update(userId); ? ? } ? }); } @Async public void update(Integer userId){ ? Icon icon = getUserIcon(userId); ? // 更新用戶圖片 ? updateUserPohot(userId,icon); }
通過將異步方法注冊為事務提交后的操作,這樣Spring可以自動幫我們在事務提交后執(zhí)行對應的操作。
參考資料
異步事務?關于異步@Async + 事務
@Transactional 事務提交 與 @Async 異步執(zhí)行
到此這篇關于SpringBoot異步與事務一起使用的問題解決的文章就介紹到這了,更多相關SpringBoot異步與事務一起使用內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
Java使用多線程批次查詢大量數據(Callable返回數據)方式
今天給大家分享Java使用多線程批次查詢大量數據(Callable返回數據)方式,多線程有好幾種方式,今天說的方式比較好,實現(xiàn)Callable<> 這種方式能返回查詢的數據,加上Future異步獲取方式,查詢效率大大加快,感興趣的朋友一起看看吧2023-11-11IDEA下Maven的pom文件導入依賴出現(xiàn)Auto build completed with errors的問題
這篇文章主要介紹了IDEA下Maven的pom文件導入依賴出現(xiàn)Auto build completed with errors,本文通過圖文并茂的形式給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-06-06聊聊Controller中RequestMapping的作用
這篇文章主要介紹了Controller中RequestMapping的作用,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-02-02