今天要討論的是「Java實現多線程單條數據事務管理」,在此以前,順便回顧一下實現多線程的幾種方式html
第一種方法是繼承Thread類,重寫run()
方法多線程
public class TestThread extends Thread { public void run() { System.out.println("繼承Thread類,重寫run方法"); } }
使用時,new一個實例,執行start()
方法併發
TestThread testThread1 = new TestThread(); // 新建狀態 TestThread testThread2 = new TestThread(); // 新建狀態 testThread1.start(); // 就緒狀態 testThread2.start(); // 就緒狀態
什麼時候執行取決於cpu調度app
由於Java「單繼承、多實現」的特性,當咱們已經繼承了一個類的時候,則沒法再繼承Thread類,此時能夠經過實現Runnable接口的方式,實現run()
方法ide
public class TestThread extends FatherClass implements Runnable { public void run() { System.out.println("實現Runnable接口的方式,實現run方法"); } }
Thread類也是實現Runnable接口測試
使用時,須要首先實例化一個Thread,並傳入本身的TestThread實例this
TestThread testThread = new TestThread(); Thread thread = new Thread(testThread); thread.start();
該方法區別於前兩種的特色是:可以得到線程處理的結果。所以該方式適用於須要對線程的結果進行處理的場景線程
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; } }
使用時,先建立TestCallable對象,而後使用FutureTask來包裝MyCallable對象,再將FutureTask對象做爲Thread對象的target建立新的線程,最後thread執行start()
方法,線程進入就緒狀態code
Callable<Integer> testCallable = new TestCallable(); // 建立TestCallable對象 FutureTask<Integer> futureTask = new FutureTask<Integer>(testCallable); // 使用FutureTask來包裝MyCallable對象 Thread thread = new Thread(futureTask); // FutureTask對象做爲Thread對象的target建立新的線程 thread.start();
關於事務,能夠看看另一篇文章談談Java事務htm
咱們有時會遇到這樣的場景:要對大批量的數據進行更新或插入操做,須要開啓多線程來提升效率,又但願每一個線程在的處理一批數據時,可以對其中每條數據進行處理的時,作到出錯時實現單條數據回滾,而不是全部數回滾(全部數據回滾後續討論)。先看代碼:
根據以上多線程知識,咱們先定義一個業務線程類以下:
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("線程{}結束",Thread.currentThread().getName()); } }
insOneBank()
方法以下,注意的@Transactional
註解的事務隔離等級爲:REQUIRES_NEW,建立一個新的事務。
@Transactional(propagation = Propagation.REQUIRES_NEW) public void insOneBank(BalBankDictEntity balBankDictEntity){ balBankDictMapper.insert(balBankDictEntity); /* 模擬發生異常,拋出異常,實現將已插入數據回滾 */ if (Integer.parseInt(balBankDictEntity.getBankId().substring(2)) % 100 == 0){ throw new BusiException("test"); } }
開啓多線程進行業務處理,注意加上@Transactional
註解
@Transactional public void testTransactional(){ /* 模擬測試數據 */ 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; // 計算每一個線程處理的數量 for (int i = 1; i <= 10 ; i++ ){ List<BalBankDictEntity> balBankDictEntityList = splitDataList(balBankDictEntities,dealNum,10,i); // 切割數據集實現數據隔離 TestTranstionalThread testTranstional = new TestTranstionalThread(balBankDictEntityList); fixedThreadPool.execute(testTranstional); } }
最終實現多個線程併發插入數據,有異常的數據的單獨回滾,不影響總體