今天要讨论的是“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多线程事务管理的更多相关文章

  1. java spring事务管理相关

    一般项目结构为: 数据持久层dao     业务层service     控制层controller 事务控制是在业务层service起作用的,所以需要同时对多张表做添加,修改或删除操作时应该在ser ...

  2. JAVA JDBC(存储过程和事务管理)

    1.什么是存储过程 存储过程(Stored Procedure)是在大型数据库系统中,一组为了完成特定功能的SQL 语句集,存储在数据库中,经过第一次编译后再次调用不需要再次编译,用户通过指定存储过程 ...

  3. 跟我学Spring3(9.2):Spring的事务之事务管理器

    原文出处: 张开涛9.2.1 概述 Spring框架支持事务管理的核心是事务管理器抽象,对于不同的数据访问框架(如Hibernate)通过实现策略接口PlatformTransactionManage ...

  4. 【Java EE 学习 54】【OA项目第一天】【SSH事务管理不能回滚问题解决】【struts2流程回顾】

    一.SSH整合之后事务问题和总结 1.引入问题:DAO层测试 假设将User对象设置为懒加载模式,在dao层使用load方法. 注意,注释不要放开. 使用如下的代码块进行测试: 会报错:no sess ...

  5. 【Java EE 学习 52】【Spring学习第四天】【Spring与JDBC】【JdbcTemplate创建的三种方式】【Spring事务管理】【事务中使用dbutils则回滚失败!!!??】

    一.JDBC编程特点 静态代码+动态变量=JDBC编程. 静态代码:比如所有的数据库连接池 都实现了DataSource接口,都实现了Connection接口. 动态变量:用户名.密码.连接的数据库. ...

  6. Java数据库连接——JDBC调用存储过程,事务管理和高级应用

    一.JDBC常用的API深入详解及存储过程的调用 相关链接:Jdbc调用存储过程 1.存储过程(Stored Procedure)的介绍 我们常用的操作数据库语言SQL语句在执行的时候需要先编译,然后 ...

  7. java事务管理

    一.什么是Java事务 通常的观念认为,事务仅与数据库相关. 事务必须服从ISO/IEC所制定的ACID原则.ACID是原子性(atomicity).一致性(consistency).隔离性(isol ...

  8. Java数据库连接--JDBC调用存储过程,事务管理和高级应用

    相关链接:Jdbc调用存储过程 一.JDBC常用的API深入详解及存储过程的调用 1.存储过程的介绍 我们常用的操作数据库语言SQL语句在执行的时候要先进行编译,然后执行,而存储过程是在大型数据库系统 ...

  9. Java框架spring Boot学习笔记(六):Spring Boot事务管理

    SpringBoot和Java框架spring 学习笔记(十九):事务管理(注解管理)所讲的类似,使用@Transactional注解便可以轻松实现事务管理.

随机推荐

  1. 『动善时』JMeter基础 — 30、JMeter中JSON断言详解

    目录 1.JSON断言组件界面详解 2.JSON断言组件的使用 (1)测试计划内包含的元件 (2)登陆接口请求界面内容 (3)JSON断言界面内容 (4)查看运行结果 (5)断言结果组件说明 3.JS ...

  2. python记录日志

    import logging from logging import handlers# 日志记录 class Logger(object): level_relations = { 'debug': ...

  3. FFmpeg扩展开发

    FFmpeg扩展开发 对FFmpeg RTMP/FLV部分做了扩展,用于支持H.265. 针对<video_file_format_spec_v10_1> VIDEODATA部分扩展如下: ...

  4. 那些年,我们一起做过的KNX智能化控制经典案例!

    那些年,我们一起做过的KNX经典案例! 光阴之箭已经穿越年轮,抵达2021 GVS在2008年加入KNX国际协会,成为中国首批引进KNX标准的企业,此后,还率先研发出基于KNX的核心协议栈,定标准,做 ...

  5. springmvc——CharacterEncodingFilter过滤器要放在所有过滤器前面

    CharacterEncodingFilter的拦截顺序必须是第一个,否则还是会出现乱码问题.这是因为 request对象的parameter并不是一开始就解析的,它是等你第一次调用getParame ...

  6. 二、Nginx 服务器升级

      1,编译新版本的nginx 软件 [root@client lnmp_soft]# tar -xf nginx-1.12.2.tar.gz -C .. [root@client lnmp_soft ...

  7. 2、java基础语法(上):变量与运算符

    关键字与保留字 关键字 定义:被Java语言赋予了特殊含义,用做专门用途的字符串(单词) 特点:关键字中所有字母都为小写 官方地址:https://docs.oracle.com/javase/tut ...

  8. 32.qt quick-模仿QQ登录界面实现3D旋转(Rotation、Flipable)

    要想模仿QQ登录界面的3D旋转,我们需要学习Rotation和Flipable.由于没找到QQ的资源图,所以我们以两个图片为例模仿QQ的3D旋转,如下图所示: 最终效果如下所示: 1.Rotation ...

  9. 玩转STM32MP157- 在应用层中使用 fbtft

    fbtft使用的是framebuffer框架,这种框架将显示设备抽象为帧缓冲区,对framebuffer设备(/dev/fbx(0.1.2..))进行相关操作可以反应到LCD上. 现在尝试下在用户空间 ...

  10. Redmine部署

    Redmine部署文章: 第一篇:Redmine部署 第二篇:Redmine部署中遇到的问题 部门内部需要项目开发维护的网站,这种网站有付费的,也有开源项目.这类项目管理与协作的工具主要的MS Sha ...