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