Java多線程事務(wù)管理的實(shí)現(xiàn)
今天要討論的是“Java實(shí)現(xiàn)多線程單條數(shù)據(jù)事務(wù)管理”,在此之前,順便回顧一下實(shí)現(xiàn)多線程的幾種方式
實(shí)現(xiàn)多線程的三種方式
一、繼承Thread類(lèi)
第一種方法是繼承Thread類(lèi),重寫(xiě)run()方法
public class TestThread extends Thread { public void run() { System.out.println("繼承Thread類(lèi),重寫(xiě)run方法"); } }
使用時(shí),new一個(gè)實(shí)例,執(zhí)行start()方法
TestThread testThread1 = new TestThread(); // 新建狀態(tài) TestThread testThread2 = new TestThread(); // 新建狀態(tài) testThread1.start(); // 就緒狀態(tài) testThread2.start(); // 就緒狀態(tài)
何時(shí)執(zhí)行取決于cpu調(diào)度
二、實(shí)現(xiàn)Runnable接口
因?yàn)镴ava“單繼承、多實(shí)現(xiàn)”的特性,當(dāng)我們已經(jīng)繼承了一個(gè)類(lèi)的時(shí)候,則無(wú)法再繼承Thread類(lèi),此時(shí)可以通過(guò)實(shí)現(xiàn)Runnable接口的方式,實(shí)現(xiàn)run()方法
public class TestThread extends FatherClass implements Runnable { public void run() { System.out.println("實(shí)現(xiàn)Runnable接口的方式,實(shí)現(xiàn)run方法"); } }
Thread類(lèi)也是實(shí)現(xiàn)Runnable接口
使用時(shí),需要首先實(shí)例化一個(gè)Thread,并傳入自己的TestThread實(shí)例
TestThread testThread = new TestThread(); Thread thread = new Thread(testThread); thread.start();
三、實(shí)現(xiàn)Callable和Future接口
該方法區(qū)別于前兩種的特點(diǎn)是:能夠獲得線程處理的結(jié)果。因此該方式適用于需要對(duì)線程的結(jié)果進(jìn)行處理的場(chǎng)景
class TestCallable implements Callable<Integer> { @Override public Integer call() { int sum = 0; for (int i = 0; i < 100; i++) { System.out.println(Thread.currentThread().getName() + " " + i); sum += i; } return sum; } }
使用時(shí),先創(chuàng)建TestCallable對(duì)象,然后使用FutureTask來(lái)包裝MyCallable對(duì)象,再將FutureTask對(duì)象作為T(mén)hread對(duì)象的target創(chuàng)建新的線程,最后thread執(zhí)行start()方法,線程進(jìn)入就緒狀態(tài)
Callable<Integer> testCallable = new TestCallable(); // 創(chuàng)建TestCallable對(duì)象 FutureTask<Integer> futureTask = new FutureTask<Integer>(testCallable); // 使用FutureTask來(lái)包裝MyCallable對(duì)象 Thread thread = new Thread(futureTask); // FutureTask對(duì)象作為T(mén)hread對(duì)象的target創(chuàng)建新的線程 thread.start();
多線程單條數(shù)據(jù)事務(wù)管理
我們有時(shí)會(huì)遇到這樣的場(chǎng)景:要對(duì)大批量的數(shù)據(jù)進(jìn)行更新或插入操作,需要開(kāi)啟多線程來(lái)提高效率,又希望每個(gè)線程在的處理一批數(shù)據(jù)時(shí),能夠?qū)ζ渲忻織l數(shù)據(jù)進(jìn)行處理的時(shí),做到出錯(cuò)時(shí)實(shí)現(xiàn)單條數(shù)據(jù)回滾,而不是所有數(shù)回滾(所有數(shù)據(jù)回滾后續(xù)討論)。先看代碼:
根據(jù)以上多線程知識(shí),我們先定義一個(gè)業(yè)務(wù)線程類(lèi)如下:
public class TestTranstionalThread extends Thread { private List<BalBankDictEntity> balBankDictEntities; public TestTranstionalThread( List<BalBankDictEntity> balBankDictEntities){ this.balBankDictEntities = balBankDictEntities; } @Override public void run() { log.info("線程{}開(kāi)始",Thread.currentThread().getName()); for (BalBankDictEntity balBankDictEntity : balBankDictEntities) { try{ collBillDao.insOneBank(balBankDictEntity); }catch (BusiException e){ log.error("{}回滾",balBankDictEntity.getBankId()); } } log.info("線程{}結(jié)束",Thread.currentThread().getName()); } }
insOneBank()方法如下,注意的@Transactional注解的事務(wù)隔離等級(jí)為:REQUIRES_NEW,創(chuàng)建一個(gè)新的事務(wù)。
@Transactional(propagation = Propagation.REQUIRES_NEW) public void insOneBank(BalBankDictEntity balBankDictEntity){ balBankDictMapper.insert(balBankDictEntity); /* 模擬發(fā)生異常,拋出異常,實(shí)現(xiàn)將已插入數(shù)據(jù)回滾 */ if (Integer.parseInt(balBankDictEntity.getBankId().substring(2)) % 100 == 0){ throw new BusiException("test"); } }
開(kāi)啟多線程進(jìn)行業(yè)務(wù)處理,注意加上@Transactional注解
@Transactional public void testTransactional(){ /* 模擬測(cè)試數(shù)據(jù) */ List<BalBankDictEntity> balBankDictEntities = new ArrayList<>(); for (int i = 0 ; i < 100000 ; i ++){ BalBankDictEntity balBankDictEntity = new BalBankDictEntity(); balBankDictEntity.setBankCode("BK" + i); balBankDictEntity.setBankId("ID" + i + ""); balBankDictEntity.setBankName("N" + i + "N"); balBankDictEntities.add(balBankDictEntity); } int totalNum = balBankDictEntities.size(); log.info("totalNum" + totalNum); /* 分10個(gè)線程處理 */ ExecutorService fixedThreadPool = Executors.newFixedThreadPool(10); int dealNum = totalNum % 10 == 0 ? totalNum / 10 : totalNum / 10 + 1; // 計(jì)算每個(gè)線程處理的數(shù)量 for (int i = 1; i <= 10 ; i++ ){ List<BalBankDictEntity> balBankDictEntityList = splitDataList(balBankDictEntities,dealNum,10,i); // 切割數(shù)據(jù)集實(shí)現(xiàn)數(shù)據(jù)隔離 TestTranstionalThread testTranstional = new TestTranstionalThread(balBankDictEntityList); fixedThreadPool.execute(testTranstional); } }
最終實(shí)現(xiàn)多個(gè)線程并發(fā)插入數(shù)據(jù),有異常的數(shù)據(jù)的單獨(dú)回滾,不影響整體
到此這篇關(guān)于Java多線程事務(wù)管理的實(shí)現(xiàn)的文章就介紹到這了,更多相關(guān)Java多線程事務(wù)管理內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- Java多線程實(shí)現(xiàn)異步調(diào)用的方法
- 詳解三種java實(shí)現(xiàn)多線程的方式
- Java多線程的實(shí)現(xiàn)方式比較(兩種方式比較)
- java實(shí)現(xiàn)多線程之定時(shí)器任務(wù)
- Java編程實(shí)現(xiàn)多線程TCP服務(wù)器完整實(shí)例
- java多線程實(shí)現(xiàn)下載圖片并壓縮
- Java多線程定時(shí)器Timer原理及實(shí)現(xiàn)
- Java多線程實(shí)現(xiàn)聊天客戶端和服務(wù)器
- Java多線程編程實(shí)現(xiàn)socket通信示例代碼
- Java多線程之Callable接口的實(shí)現(xiàn)
相關(guān)文章
Java?精煉解讀數(shù)據(jù)結(jié)構(gòu)的鏈表的概念與實(shí)現(xiàn)
鏈表是一種物理存儲(chǔ)單元上非連續(xù)、非順序的存儲(chǔ)結(jié)構(gòu),數(shù)據(jù)元素的邏輯順序是通過(guò)鏈表中的指針連接次序?qū)崿F(xiàn)的,每一個(gè)鏈表都包含多個(gè)節(jié)點(diǎn),節(jié)點(diǎn)又包含兩個(gè)部分,一個(gè)是數(shù)據(jù)域,一個(gè)是引用域2022-03-03Elasticsearch學(xué)習(xí)之Terms?set?查詢(xún)
這篇文章主要為大家介紹了Elasticsearch學(xué)習(xí)Terms?set?查詢(xún)示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-02-02小白也可以學(xué)會(huì)的Java NIO的Write事件
剛開(kāi)始對(duì)NIO的寫(xiě)操作理解的不深,不知道為什么要注冊(cè)寫(xiě)事件,何時(shí)注冊(cè)寫(xiě)事件,為什么寫(xiě)完之后要取消注冊(cè)寫(xiě)事件,今天特地整理了本篇文章,需要的朋友可以參考下2021-06-06Java使用poi組件導(dǎo)出Excel格式數(shù)據(jù)
這篇文章主要介紹了Java使用poi組件導(dǎo)出Excel格式數(shù)據(jù),需要的朋友可以參考下2020-02-02Java SpringBoot啟動(dòng)指定profile的8種方式詳解
這篇文章主要介紹了spring boot 如何指定profile啟動(dòng)的操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-09-09基于Java語(yǔ)言在窗體上實(shí)現(xiàn)飛機(jī)大戰(zhàn)小游戲的完整步驟
這篇文章主要給大家介紹了基于Java語(yǔ)言在窗體上實(shí)現(xiàn)飛機(jī)大戰(zhàn)小游戲的完整步驟,文中通過(guò)圖文以及實(shí)例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2022-02-02java的Jackson將json字符串轉(zhuǎn)換成泛型List
這篇文章主要介紹了java的Jackson將json字符串轉(zhuǎn)換成泛型List ,這里整理了詳細(xì)的代碼,有需要的小伙伴可以參考下。2017-02-02java獲取指定開(kāi)始時(shí)間與結(jié)束時(shí)間之間的所有日期
這篇文章主要為大家詳細(xì)介紹了java獲取指定開(kāi)始時(shí)間與結(jié)束時(shí)間之間的所有日期,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-05-05Java使用正則表達(dá)式去除小數(shù)點(diǎn)后面多余的0功能示例
這篇文章主要介紹了Java使用正則表達(dá)式去除小數(shù)點(diǎn)后面多余的0功能,結(jié)合具體實(shí)例形式分析了java字符串正則替換相關(guān)操作技巧,需要的朋友可以參考下2017-06-06