您好,登錄后才能下訂單哦!
Java中怎么實現(xiàn)多線程事務(wù)管理,針對這個問題,這篇文章詳細介紹了相對應(yīng)的分析和解答,希望可以幫助更多想解決這個問題的小伙伴找到更簡單易行的方法。
第一種方法是繼承Thread類,重寫run()方法
public class TestThread extends Thread { public void run() { System.out.println("繼承Thread類,重寫run方法"); } }
使用時,new一個實例,執(zhí)行start()方法
TestThread testThread1 = new TestThread(); // 新建狀態(tài) TestThread testThread2 = new TestThread(); // 新建狀態(tài) testThread1.start(); // 就緒狀態(tài) testThread2.start(); // 就緒狀態(tài)
何時執(zhí)行取決于cpu調(diào)度
因為Java“單繼承、多實現(xiàn)”的特性,當我們已經(jīng)繼承了一個類的時候,則無法再繼承Thread類,此時可以通過實現(xiàn)Runnable接口的方式,實現(xiàn)run()方法
public class TestThread extends FatherClass implements Runnable { public void run() { System.out.println("實現(xiàn)Runnable接口的方式,實現(xiàn)run方法"); } }
Thread類也是實現(xiàn)Runnable接口
使用時,需要首先實例化一個Thread,并傳入自己的TestThread實例
TestThread testThread = new TestThread(); Thread thread = new Thread(testThread); thread.start();
該方法區(qū)別于前兩種的特點是:能夠獲得線程處理的結(jié)果。因此該方式適用于需要對線程的結(jié)果進行處理的場景
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; } }
使用時,先創(chuàng)建TestCallable對象,然后使用FutureTask來包裝MyCallable對象,再將FutureTask對象作為Thread對象的target創(chuàng)建新的線程,最后thread執(zhí)行start()方法,線程進入就緒狀態(tài)
Callable<Integer> testCallable = new TestCallable(); // 創(chuàng)建TestCallable對象 FutureTask<Integer> futureTask = new FutureTask<Integer>(testCallable); // 使用FutureTask來包裝MyCallable對象 Thread thread = new Thread(futureTask); // FutureTask對象作為Thread對象的target創(chuàng)建新的線程 thread.start();
我們有時會遇到這樣的場景:要對大批量的數(shù)據(jù)進行更新或插入操作,需要開啟多線程來提高效率,又希望每個線程在的處理一批數(shù)據(jù)時,能夠?qū)ζ渲忻織l數(shù)據(jù)進行處理的時,做到出錯時實現(xiàn)單條數(shù)據(jù)回滾,而不是所有數(shù)回滾(所有數(shù)據(jù)回滾后續(xù)討論)。先看代碼:
根據(jù)以上多線程知識,我們先定義一個業(yè)務(wù)線程類如下:
public class TestTranstionalThread extends Thread { private List<BalBankDictEntity> balBankDictEntities; public TestTranstionalThread( List<BalBankDictEntity> balBankDictEntities){ this.balBankDictEntities = balBankDictEntities; } @Override public void run() { log.info("線程{}開始",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ù)隔離等級為:REQUIRES_NEW,創(chuàng)建一個新的事務(wù)。
@Transactional(propagation = Propagation.REQUIRES_NEW) public void insOneBank(BalBankDictEntity balBankDictEntity){ balBankDictMapper.insert(balBankDictEntity); /* 模擬發(fā)生異常,拋出異常,實現(xiàn)將已插入數(shù)據(jù)回滾 */ if (Integer.parseInt(balBankDictEntity.getBankId().substring(2)) % 100 == 0){ throw new BusiException("test"); } }
開啟多線程進行業(yè)務(wù)處理,注意加上@Transactional注解
@Transactional public void testTransactional(){ /* 模擬測試數(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個線程處理 */ ExecutorService fixedThreadPool = Executors.newFixedThreadPool(10); int dealNum = totalNum % 10 == 0 ? totalNum / 10 : totalNum / 10 + 1; // 計算每個線程處理的數(shù)量 for (int i = 1; i <= 10 ; i++ ){ List<BalBankDictEntity> balBankDictEntityList = splitDataList(balBankDictEntities,dealNum,10,i); // 切割數(shù)據(jù)集實現(xiàn)數(shù)據(jù)隔離 TestTranstionalThread testTranstional = new TestTranstionalThread(balBankDictEntityList); fixedThreadPool.execute(testTranstional); } }
關(guān)于Java中怎么實現(xiàn)多線程事務(wù)管理問題的解答就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,如果你還有很多疑惑沒有解開,可以關(guān)注億速云行業(yè)資訊頻道了解更多相關(guān)知識。
免責聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。