91超碰碰碰碰久久久久久综合_超碰av人澡人澡人澡人澡人掠_国产黄大片在线观看画质优化_txt小说免费全本

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Java中怎么實現多線程事務管理

發布時間:2021-07-14 11:13:45 來源:億速云 閱讀:374 作者:Leah 欄目:開發技術

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);

    }
}

關于Java中怎么實現多線程事務管理問題的解答就分享到這里了,希望以上內容可以對大家有一定的幫助,如果你還有很多疑惑沒有解開,可以關注億速云行業資訊頻道了解更多相關知識。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

武功县| 东源县| 天峻县| 陆河县| 屏东市| 梓潼县| 磴口县| 柏乡县| 哈密市| 喀什市| 连云港市| 贺州市| 太康县| 江源县| 延川县| 昌乐县| 临猗县| 青州市| 会理县| 门源| 枞阳县| 潞城市| 都昌县| 左贡县| 砚山县| 平凉市| 祥云县| 四川省| 惠来县| 高密市| 漯河市| 临海市| 互助| 图木舒克市| 汉川市| 沁阳市| 浑源县| 宁阳县| 东城区| 东宁县| 中江县|