Java多线程事务管理的实现

作者:Acelin_H''s Blog 时间:2023-07-26 22:58:20 

目录
  • 实现多线程的三种方式

    • 一、继承Thread类

    • 二、实现Runnable接口

    • 三、实现Callable和Future接口

  • 多线程单条数据事务管理

    今天要讨论的是“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);

    }
    }

    最终实现多个线程并发插入数据,有异常的数据的单独回滚,不影响整体

    来源:https://www.cnblogs.com/acelin/p/15003247.html

    标签:Java,多线程,事务管理
    0
    投稿

    猜你喜欢

  • 超酷炫的Android碎纸机效果推荐

    2022-05-21 01:20:44
  • RestTemplate在Spring或非Spring环境下使用精讲

    2022-03-03 07:35:41
  • SpringBoot在RequestBody中使用枚举参数案例详解

    2022-12-15 05:16:30
  • C#制作鹰眼的详细全过程(带注释)实例代码

    2022-03-01 06:56:12
  • Android自定义星星可滑动评分控件

    2022-03-18 10:23:03
  • SSh结合Easyui实现Datagrid的分页显示

    2022-01-14 21:46:49
  • 浅谈java中==以及equals方法的用法

    2021-09-29 15:21:45
  • Android提高之MediaPlayer音视频播放

    2021-05-29 08:44:37
  • C#计算两个文件的相对目录算法的实例代码

    2022-08-27 10:27:44
  • C#6.0新语法示例详解

    2023-11-16 03:43:42
  • java dump文件怎么生成和分析-JMAP用法详解

    2021-06-03 23:59:43
  • spring boot ${}占位符不起作用的解决方案

    2022-06-28 20:49:29
  • Android Studio编写AIDL文件后如何实现自动编译生成

    2022-01-25 16:44:38
  • Android树形控件绘制方法

    2022-09-05 01:50:55
  • 如何解决Java多线程死锁问题

    2022-08-11 15:51:02
  • java设计模式-单例模式实现方法详解

    2022-02-15 13:56:01
  • Android通过访问网页查看网页源码实例详解

    2023-10-05 03:09:11
  • Flutter基于Dart Unwrapping Multiple Optional小技巧

    2023-07-05 11:44:19
  • Android Scroller的使用方法

    2023-02-03 03:57:01
  • Java计算文本MD5加密值的方法示例

    2023-11-15 13:18:48
  • asp之家 软件编程 m.aspxhome.com