Java使用多线程异步执行批量更新操作方法

写在前面:

相信不少开发者在遇到项目对数据进行批量操作的时候,都会有不少的烦恼,尤其是针对数据量极大的情况下,效率问题就直接提上了菜板。因此,开多线程来执行批量任务是十分重要的一种批量操作思路,其实这种思路实现起来也十分简单,就拿批量更新的操作举例:

整体流程图

  • 步骤 获取需要进行批量更新的大集合A,对大集合进行拆分操作,分成N个小集合A-1 ~ A-N 。
  • 开启线程池,针对集合的大小进行调参,对小集合进行批量更新操作。
  • 对流程进行控制,控制线程执行顺序。 按照指定大小拆分集合的工具类
import com.google.common.collect.Lists;
import org.apache.commons.collections.CollectionUtils;

import java.util.List;

/**
 * 拆分结合工具类
 *
 * @author shiwen
 * @date 2020/12/27
 */
public class SplitListUtils {

 /**
  * 拆分集合
  *
  * @param <T> 泛型对象
  * @param resList 需要拆分的集合
  * @param subListLength 每个子集合的元素个数
  * @return 返回拆分后的各个集合组成的列表
  * 代码里面用到了guava和common的结合工具类
  **/
 public static <T> List<List<T>> split(List<T> resList, int subListLength) {
  if (CollectionUtils.isEmpty(resList) || subListLength <= 0) {
   return Lists.newArrayList();
  }
  List<List<T>> ret = Lists.newArrayList();
  int size = resList.size();
  if (size <= subListLength) {
   // 数据量不足 subListLength 指定的大小
   ret.add(resList);
  } else {
   int pre = size / subListLength;
   int last = size % subListLength;
   // 前面pre个集合,每个大小都是 subListLength 个元素
   for (int i = 0; i < pre; i++) {
    List<T> itemList = Lists.newArrayList();
    for (int j = 0; j < subListLength; j++) {
     itemList.add(resList.get(i * subListLength + j));
    }
    ret.add(itemList);
   }
   // last的进行处理
   if (last > 0) {
    List<T> itemList = Lists.newArrayList();
    for (int i = 0; i < last; i++) {
     itemList.add(resList.get(pre * subListLength + i));
    }
    ret.add(itemList);
   }
  }
  return ret;
 }

 // 运行代码
 public static void main(String[] args) {
  List<String> list = Lists.newArrayList();
  int size = 1099;
  for (int i = 0; i < size; i++) {
   list.add("hello-" + i);
  }
  // 大集合里面包含多个小集合
  List<List<String>> temps = split(list, 100);
  int j = 0;
  // 对大集合里面的每一个小集合进行操作
  for (List<String> obj : temps) {
   System.out.println(String.format("row:%s -> size:%s,data:%s", ++j, obj.size(), obj));
  }
 }

}

开启异步执行任务的线程池

public void threadMethod() {
 List<T> updateList = new ArrayList();
 // 初始化线程池, 参数一定要一定要一定要调好!!!!
 ThreadPoolExecutor threadPool = new ThreadPoolExecutor(20, 50,
   4, TimeUnit.SECONDS, new ArrayBlockingQueue(10), new ThreadPoolExecutor.AbortPolicy());
 // 大集合拆分成N个小集合, 这里集合的size可以稍微小一些(这里我用100刚刚好), 以保证多线程异步执行, 过大容易回到单线程
 List<T> splitNList = SplitListUtils.split(totalList, 100);
 // 记录单个任务的执行次数
 CountDownLatch countDownLatch = new CountDownLatch(splitNList.size());
 // 对拆分的集合进行批量处理, 先拆分的集合, 再多线程执行
 for (List<T> singleList : splitNList) {
  // 线程池执行
  threadPool.execute(new Thread(new Runnable(){
   @Override
   public void run() {
    for (Entity yangshiwen : singleList) {
     // 将每一个对象进行数据封装, 并添加到一个用于存储更新数据的list
     // ......

     // 任务个数 - 1, 直至为0时唤醒await()
     countDownLatch.countDown();
    }
   }
  }));
 }
 try {
  // 让当前线程处于阻塞状态,直到锁存器计数为零
  countDownLatch.await();
 } catch (InterruptedException e) {
  throw new BusinessLogException(ResponseEnum.FAIL);
 }
 // 通过mybatis的批量插入的方式来进行数据的插入, 这一步还是要做判空
 if (GeneralUtil.listNotNull(updateList)) {
  batchUpdateEntity(updateList);
  LogUtil.info("xxxxxxxxxxxxxxx");
 }
}

写在最后

多线程是Java的一个难点,但是它也很有趣,听说玩得溜得起飞的人,人生都开启多线程模式了…

到此这篇关于Java使用多线程异步执行批量更新操作的文章就介绍到这了,更多相关java多线程异步执行批量更新内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • Java多线程批量数据导入的方法详解

    前言: 当遇到大量数据导入时,为了提高处理的速度,可以选择使用多线程来批量处理这些处理.常见的场景有: 大文件导入数据库(这个文件不一定是标准的CSV可导入文件或者需要在内存中经过一定的处理) 数据同步(从第三方接口拉取数据处理后写入自己的数据库) 以上的场景有一个共性,这类数据导入的场景简单来说就是将数据从一个数据源移动到另外一个数据源,而其中必定可以分为两步 数据读取:从数据源读取数据到内存 数据写入:将内存中的数据写入到另外一个数据源,可能存在数据处理 而且根据读取的速度一般会比数据写入的

  • Java多线程下解决资源竞争的7种方法详解

    前言 一般情况下,只要涉及到多线程编程,程序的复杂性就会显著上升,性能显著下降,BUG出现的概率大大提升. 多线程编程本意是将一段程序并行运行,提升数据处理能力,但是由于大部分情况下都涉及到共有资源的竞争,所以修改资源 对象时必须加锁处理.但是锁的实现有很多种方法,下面就来一起了解一下在C#语言中几种锁的实现与其性能表现. 一.c#下的几种锁的运用方式 1.临界区,通过对多线程的串行化来访问公共资源或一段代码,速度快,适合控制数据访问. private static object obj = n

  • Java创建多线程异步执行实现代码解析

    实现Runable接口 通过实现Runable接口中的run()方法 public class ThreadTest implements Runnable { public static void main(String[] args) { Thread thread = new Thread(new ThreadTest()); thread.start(); } @Override public void run() { System.out.println("Runable 方式创建的新

  • Java使用多线程异步执行批量更新操作方法

    写在前面: 相信不少开发者在遇到项目对数据进行批量操作的时候,都会有不少的烦恼,尤其是针对数据量极大的情况下,效率问题就直接提上了菜板.因此,开多线程来执行批量任务是十分重要的一种批量操作思路,其实这种思路实现起来也十分简单,就拿批量更新的操作举例: 整体流程图 步骤 获取需要进行批量更新的大集合A,对大集合进行拆分操作,分成N个小集合A-1 ~ A-N . 开启线程池,针对集合的大小进行调参,对小集合进行批量更新操作. 对流程进行控制,控制线程执行顺序. 按照指定大小拆分集合的工具类 impo

  • mybatis执行批量更新batch update 的方法(oracle,mysql两种)

    Oracle和MySQL数据库的批量update在mybatis中配置不太一样: oracle数据库: <code class="hljs tcl" style=""><<span class="hljs-keyword" style="">update</span> id=<span class="hljs-string" style=""

  • DataAdapter执行批量更新的实例代码

    在以前版本的 ADO.NET 中,使用 DataSet 中的更改来更新数据库时,DataAdapter 的 Update 方法每次更新数据库的一行.因为该方法循环访问指定 DataTable 中的行,所以,会检查每个 DataRow,确定是否已修改.如果该行已修改,将根据该行的 RowState 属性值调用相应的 UpdateCommand.InsertCommand 或 DeleteCommand.每一次行更新都涉及网络与数据库之间的双向数据传输.    在 ADO.NET 2.0 中,Dat

  • C#多线程异步执行和跨线程访问控件Helper

    一.工具类代码 public class TaskHelper { #region 多线程操作 /// <summary> /// 功能描述:多线程执行方法,方法无参数,无返回值 /// </summary> /// <param name="func">方法,如果方法中调用了控件,请使用 ThreadInvokerControl(() => { 您的操作})进行包括</param> /// <param name="

  • Mybatis批量更新报错问题

    下面给大家介绍mybatis批量更新报错问题, allowMultiQueries=true 后来发现是jdbc链接没有加允许批量更新操作的参数引起的,不加会报badsql,mysql版的mybatis批量更新操作如下 <update id="updateOrderOverdueStatus" parameterType="java.util.List"> <foreach collection="list" item=&quo

  • 在ASP.NET 2.0中操作数据之三十七:DataList批量更新

    导言 在前面我们学习了如何创建item级的DataList.和可编辑的GridView一样,每个DataList里的item都包含一个Edit button,当点击时,item会变的可编辑.item级的编辑在偶尔需要更新时没什么问题,但是在有些情况下用户需要编辑大量的记录.如果一个用户需要编辑许多记录,他会被迫去不停的去点击Edit,作出修改,然后点击Update,这些大量的点击会妨碍他的工作效率.在这样的情况下,一个好的选择是提供一个完全可编辑的DataList,它的所有的item都处于编辑模

  • jsp Hibernate批量更新和批量删除处理代码

    以下程序直接通过Hibernate API批量更新CUSTOMERS表中年龄大于零的所有记录的AGE字段: tx = session.beginTransaction();Iterator customers=session.find("from Customer c where c.age>0").iterator();while(customers.hasNext()){Customer customer=(Customer)customers.next();customer

  • Java 多线程有序执行的几种方法总结

    Java 多线程有序执行的几种方法总结 同事无意间提出了这个问题,亲自实践了两种方法.当然肯定还会有更多更好的方法. 方法一 import java.util.concurrent.atomic.AtomicInteger; public class OrderedThread1 { static AtomicInteger count = new AtomicInteger(0); public static void main(String[] args) throws Interrupte

  • 浅谈Java中spring 线程异步执行

    多线程并发处理起来通常比较麻烦,如果你使用spring容器来管理业务bean,事情就好办了多了.spring封装了Java的多线程的实现,你只需要关注于并发事物的流程以及一些并发负载量等特性,具体来说如何使用spring来处理并发事务: 1.了解 TaskExecutor接口 Spring的TaskExecutor接口等同于java.util.concurrent.Executor接口. 实际上,它存在的主要原因是为了在使用线程池的时候,将对Java5的依赖抽象出来. 这个接口只有一个方法exe

随机推荐