欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java多線程事務(wù)管理的實(shí)現(xiàn)

 更新時(shí)間:2021年07月14日 08:50:45   作者:Acelin_H''''s Blog  
本文主要介紹了Java多線程事務(wù)管理的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧

今天要討論的是“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)文章希望大家以后多多支持腳本之家!

相關(guān)文章

最新評(píng)論