spring 使用多線程,保證事務一致性

1、背景

最近接受到接口優化的任務,查看代碼邏輯后發現在批量處理數據耗時長,想到使用多線程處理批量數據,又要保持原來的事務一致性。

2、實現方法

(1)、創建多線程事務管理

@Component
@Slf4j
public class MultiThreadingTransactionManager {/*** 數據源事務管理器*/@Autowiredprivate DataSourceTransactionManager dataSourceTransactionManager;@Autowiredprivate ThreadPoolTaskExecutor executorService;private long timeout = 120;/*** 用于判斷子線程業務是否處理完成* 處理完成時threadCountDownLatch的值為0*/private CountDownLatch threadCountDownLatch;/*** 用于等待子線程全部完成后,子線程統一進行提交和回滾* 進行提交和回滾時mainCountDownLatch的值為0*/private final CountDownLatch mainCountDownLatch = new CountDownLatch(1);/*** 是否提交事務,默認是true,當子線程有異常發生時,設置為false,回滾事務*/private final AtomicBoolean isSubmit = new AtomicBoolean(true);public boolean execute(List<Runnable> runnableList,String factorySchema) {isSubmit.set(true);setThreadCountDownLatch(runnableList.size());runnableList.forEach(runnable -> executorService.execute(() -> executeThread(factorySchema,runnable, threadCountDownLatch, mainCountDownLatch, isSubmit)));// 等待子線程全部執行完畢try {// 若計數器變為零了,則返回 trueboolean isFinish = threadCountDownLatch.await(timeout, TimeUnit.SECONDS);if (!isFinish) {// 如果還有為執行完成的就回滾isSubmit.set(false);log.info("存在子線程在預期時間內未執行完畢,任務將全部回滾");}} catch (Exception exception) {log.info("主線程發生異常,異常為: " + exception.getMessage());} finally {// 計數器減1,代表該主線程執行完畢mainCountDownLatch.countDown();}// 返回結果,是否執行成功,事務提交即為執行成功,事務回滾即為執行失敗return isSubmit.get();}private void executeThread(String factorySchema,Runnable runnable, CountDownLatch threadCountDownLatch, CountDownLatch mainCountDownLatch, AtomicBoolean isSubmit) {log.info("子線程: [" + Thread.currentThread().getName() + "]");// 判斷別的子線程是否已經出現錯誤,錯誤別的線程已經出現錯誤,那么所有的都要回滾,這個子線程就沒有必要執行了if (!isSubmit.get()) {log.info("整個事務中有子線程執行失敗需要回滾, 子線程: [" + Thread.currentThread().getName() + "] 終止執行");// 計數器減1,代表該子線程執行完畢threadCountDownLatch.countDown();return;}//動態數據源切換SchemaContextHolder.setSchema(factorySchema);// 開啟事務DefaultTransactionDefinition defaultTransactionDefinition = new DefaultTransactionDefinition();TransactionStatus transactionStatus = dataSourceTransactionManager.getTransaction(defaultTransactionDefinition);try {// 執行業務邏輯runnable.run();} catch (Exception exception) {// 發生異常需要進行回滾,設置isSubmit為falseisSubmit.set(false);log.info("子線程: [" + Thread.currentThread().getName() + "]執行業務發生異常,異常為: " + exception.getMessage());} finally {// 計數器減1,代表該子線程執行完畢threadCountDownLatch.countDown();}try {// 等待主線程執行mainCountDownLatch.await();} catch (Exception exception) {log.info("子線程: [" + Thread.currentThread().getName() + "]等待提交或回滾異常,異常為: " + exception.getMessage());}try {// 提交if (isSubmit.get()) {dataSourceTransactionManager.commit(transactionStatus);log.info("子線程: [" + Thread.currentThread().getName() + "]進行事務提交");} else {dataSourceTransactionManager.rollback(transactionStatus);log.info("子線程: [" + Thread.currentThread().getName() + "]進行事務回滾");}} catch (Exception exception) {log.info("子線程: [" + Thread.currentThread().getName() + "]進行事務提交或回滾出現異常,異常為:" + exception.getMessage());}}private void setThreadCountDownLatch(int num) {this.threadCountDownLatch = new CountDownLatch(num);}
}

(2)、測試類

@RestController
@RequestMapping("test")
public class TestController {@AutowiredTestService testService;@AutowiredMultiThreadingTransactionManager multiThreadingTransactionManager;@RequestMapping("test")public String test(){List<TestBean> list = new ArrayList<>();list.add(new TestBean("2",1));list.add(new TestBean("3",2));List<Runnable> runnableList = new ArrayList<>();list.forEach(testBean -> runnableList.add(() -> {testService.insert(testBean);}));boolean isSuccess = multiThreadingTransactionManager.execute(runnableList,"db9771");System.out.println(isSuccess);return "ok";};
}

3、總結

大體思路,就是所有子線程在各自線程內開啟事務,執行業務邏輯后,判斷是否拋錯,一旦拋錯,會把全局AtomicBoolean置為false,因為其具有原子性所以不會有線程不安全問題。所有子線程完業務代碼會等待主線程,全部子線程執行業務結束后,主線程等待結束,判斷AtomicBoolean是什么狀態,一旦false,所有子線程回滾,否則提交。

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/web/23197.shtml
繁體地址,請注明出處:http://hk.pswp.cn/web/23197.shtml
英文地址,請注明出處:http://en.pswp.cn/web/23197.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

海外BGP服務器有什么功能?

當企業選擇海外的BGP服務器進行租用時&#xff0c;能夠實現哪些功能呢&#xff1f; 當企業擁有海外的BGP服務器時&#xff0c;可以改善網站的訪問速度&#xff0c;對于面向全球用戶的網站或者是應用來說&#xff0c;能夠通過在不同區域所部署的BGP服務器&#xff0c;用戶可以根…

【Unity Shader入門精要 第13章】使用深度和法線紋理(一)

1. 原理 深度紋理的本質是一張RenderTexture&#xff0c;只不過其中記錄的不是顏色值&#xff0c;而是一個深度值 這些深度值來自于頂點在空間變換后得到的歸一化設備坐標&#xff08;NDC&#xff09;的Z值 由于NDC坐標的分量取值范圍在[-1, 1]之間&#xff0c;要使顏色值能…

基于pytorch的車牌識別

&#x1f368; 本文為&#x1f517;365天深度學習訓練營 中的學習記錄博客&#x1f356; 原作者&#xff1a;K同學啊 一、導入數據 from torchvision.transforms import transforms from torch.utils.data import DataLoader from torchvision import datase…

RSA 非對稱加密:

非對稱加密 RSA 擁有兩個密鑰&#xff0c; 分別為 公鑰 和 私鑰&#xff0c; 服務器端擁有公鑰和私鑰&#xff0c; 二客戶端&#xff0c;只有公鑰&#xff0c; 這個公鑰可以隨便傳&#xff0c;即使被截獲也沒有關系&#xff0c; 加密使用公鑰&#xff0c; 而解密&#xff0c;…

Mysql時間操作

一、MySql時間戳轉換 select unix_timestamp(); #獲取時間戳格式時間 select FROM_UNIXTIME(1717399499); #將時間戳轉換為普通格式時間二、Mysql時間相加減結果轉換為秒 方法1&#xff1a;time_to_sec(timediff(endTime, startTime)) SELECTDISTINCT(column1),min(last_mo…

在Jenkins 中使用 NVM 管理 Node.js 部署項目的自動化腳本

在Jenkins 中使用 NVM 管理 Node.js 部署項目的自動化腳本 人生旅途&#xff0c;總有人不斷地走來&#xff0c;有人不斷地離去。當新名字變成老名字&#xff0c;當老的名字漸漸模糊&#xff0c;又是一個故事的結束和另一個故事的開始。 在現代軟件開發中&#xff0c;持續集成/持…

容器化實踐:DevOps環境下的容器交付流程

DevOps的興起是為了應對市場和消費者對技術應用的不斷增長的需求。它的目標是構建一個更快的開發環境&#xff0c;同時保持軟件的高質量標準。DevOps還致力于在敏捷開發周期中提升軟件的整體品質。這一目標的實現依賴于多種技術、平臺和工具的綜合運用。 結合容器化技術與DevO…

深入理解mysql中的各種超時屬性

1. 前言 connectTimeout: 連接超時 loginTimeout: 登錄超時 socketTimeout: Socket網絡超時&#xff0c;即讀超時 queryTimeout: sql執行超時 transactionTimeout:spring事務超時 innodb_lock_wait_timeout:innodb鎖等待超時 wait_timeout:非交互式連接關閉前的等待時間 inter…

uniapp小程序多線程 Worker 實戰【2024】

需求 最近遇到個小程序異步解碼的需求&#xff0c;采用了WebAssembly&#xff0c;涉及大量的計算。由于小程序的雙線程模型只有一個線程處理數據&#xff0c;因此智能尋求其它的解決方案。查看小程序的文檔&#xff0c;發現小程序還提供一個異步線程的Worker方案&#xff0c;可…

代碼隨想錄算法訓練營第25天|回溯

回溯part02 216. 組合總和 III /*** param {number} k* param {number} n* return {number[][]}*/ var combinationSum3 function(k, n) {// k個數字相加為n// 只能使用1-9// 每個數字只能使用一次// 不能重復 如 1 2 4 、 4 1 2 不可以let res [];backtracking(k, n, [], …

聯想Y410P跑大模型

安裝vs 2017 查看GPU版本 查看支持哪個版本的cuda windows cuda更新教程_cuda 12.0-CSDN博客 下載并安裝cuda tookit 10.1 CUDA Toolkit 10.1 Update 2 Archive | NVIDIA Developer 找到下載的文件&#xff0c;安裝 參考安裝鏈接 Win10 Vs2017 CUDA10.1安裝&#xff08;避坑…

Due to a bug fix in https://github.com/huggingface/transformers/pull/28687

錯誤&#xff1a; Due to a bug fix in https://github.com/huggingface/transformers/pull/28687 transcription using a multilingual Whisper will default to language detection followed by transcription instead of translation to English.This might be a breaking …

InnoDB存儲引擎非常重要的一個機制--MVCC(多版本并發控制)

Mysql是如何實現隔離性的&#xff1f;&#xff08;鎖MVCC&#xff09; 隔離性是指一個事務內部的操作以及操作的數據對正在進行的其他事務是隔離的&#xff0c;并發執行的各個事務之間不能相互干擾。隔離性可以防止多個事務并發執行時&#xff0c;可能存在交叉執行導致數據的不…

安全U盤和普通U盤有什么區別?

安全U盤&#xff08;也稱為加密U盤或安全閃存驅動器&#xff09;與普通U盤肯定是有一些區別的&#xff0c;從字面意思上來看&#xff0c;就能看出&#xff0c;安全U盤是能夠保護文件數據安全性的&#xff0c;普通U盤沒這一些功能的&#xff0c;可隨意拷貝文件&#xff0c;不防盜…

面試4:c++(數位物聯)

1.const 關健字的作用 定義常量&#xff0c;防止變量被意外修改&#xff0c;增強程序的可讀性和維護性。 可以用于指針&#xff0c;聲明指向常量的指針或常量指針。 2.static關健字的作用 (1)在函數內&#xff0c;用于修飾局部變量&#xff0c;使其生命周期延長到整個程序運行期…

mybatisplus多數據源內置方法報Invalid bound statement (not found)

在用mybatis-plus多數據源時用mapper內置的 selectList(queryWrapper) 查詢數據報org.apache.ibatis.binding.BindingException: Invalid bound statement (not found): 問題是在配置多數據源時用的是SqlSessionFactoryBean&#xff0c;改為MybatisSqlSessionFactoryBean即可…

Python怎么逐行處理文件:深度解析與實用技巧

Python怎么逐行處理文件&#xff1a;深度解析與實用技巧 在Python中&#xff0c;逐行處理文件是一項常見且重要的任務。無論是讀取大型日志文件、分析文本數據還是處理配置文件&#xff0c;逐行讀取都能幫助我們更有效地管理內存并提高處理速度。本文將詳細介紹Python中逐行處…

一文了解UVLED線光源的應用

在機器視覺系統中&#xff0c;光源作為不可或缺的一部分&#xff0c;能夠提高目標成像效果&#xff0c;增強檢測效果。光源的選擇至關重要&#xff0c;選到不合適的會影響成像及檢測效果。針對不同的檢測對象,不同的形狀光源應運而生。我們來看看最UVLED線光源。 下面以CCS的光…

某紅書旋轉滑塊驗證碼分析與協議算法實現

文章目錄 1. 寫在前面2. 接口分析3. 驗證軌跡4. 算法還原【??作者主頁】:吳秋霖 【??作者介紹】:擅長爬蟲與JS加密逆向分析!Python領域優質創作者、CSDN博客專家、阿里云博客專家、華為云享專家。一路走來長期堅守并致力于Python與爬蟲領域研究與開發工作! 【??作者推…

zoomeye api報錯 request invalid, validate usage and try again

項目場景&#xff1a; 調用zoomeye的api接口進行數據拿取 問題描述 之前接口一直通著今天突然報錯&#xff0c;以下為源代碼 pip install zoomeye from zoomeye.sdk import ZoomEye zm ZoomEye(api_key"34A8B452-D874-C63E0-8471-F3D4f89766f") zm.dork_search(a…