今天要討論的是“Java實現多執行緒單條資料事務管理”,在此之前,順便回顧一下實現多執行緒的幾種方式
實現多執行緒的三種方式
一、繼承Thread類
第一種方法是繼承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排程
二、實現Runnable介面
因為Java“單繼承、多實現”的特性,當我們已經繼承了一個類的時候,則無法再繼承Thread類,此時可以通過實現Runnable介面的方式,實現run()
方法
public class TestThread extends FatherClass implements Runnable {
public void run() {
System.out.println("實現Runnable介面的方式,實現run方法");
}
}
Thread類也是實現Runnable介面
使用時,需要首先例項化一個Thread,並傳入自己的TestThread例項
TestThread testThread = new TestThread();
Thread thread = new Thread(testThread);
thread.start();
三、實現Callable和Future介面
該方法區別於前兩種的特點是:能夠獲得執行緒處理的結果。因此該方式適用於需要對執行緒的結果進行處理的場景
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()
方法,執行緒進入就緒狀態
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();
多執行緒單條資料事務管理
我們有時會遇到這樣的場景:要對大批量的資料進行更新或插入操作,需要開啟多執行緒來提高效率,又希望每個執行緒在的處理一批資料時,能夠對其中每條資料進行處理的時,做到出錯時實現單條資料回滾,而不是所有數回滾(所有資料回滾後續討論)。先看程式碼:
根據以上多執行緒知識,我們先定義一個業務執行緒類如下:
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);
}
}
最終實現多個執行緒併發插入資料,有異常的資料的單獨回滾,不影響整體