Java 利用JUC CountDownLatch 线程池Executors 实现多线程操作

简介: Java 利用JUC CountDownLatch 线程池Executors 实现多线程操作业务场景:某个业务操作非常耗时,但又必须等这个操作结束后才能进行后续操作

Java 利用JUC CountDownLatch 线程池Executors 实现多线程操作

业务场景:某个业务操作非常耗时,但又必须等这个操作结束后才能进行后续操作

importorg.springframework.util.CollectionUtils;

importjava.util.List;

importjava.util.concurrent.CountDownLatch;

importjava.util.concurrent.ExecutorService;

importjava.util.concurrent.Executors;

importjava.util.concurrent.TimeUnit;

importjava.util.function.Consumer;

importjava.util.stream.Collectors;

importjava.util.stream.Stream;

/**

* 多线程任务处理工具类

* gzh-JavaNice

*/

publicclassTaskDisposeUtils {

   //并行线程数

   publicstaticfinalintPOOL_SIZE;

   static {

       //判断核心线程数 如果机器的核心线程数大于5则用机器核心线程数

       POOL_SIZE=Integer.max(Runtime.getRuntime().availableProcessors(), 5);

   }

   /**

    * 并行处理,并等待结束

    *

    * @param taskList 任务列表

    * @param consumer 消费者

    * @param <T>

    * @throws InterruptedException

    */

   publicstatic<T>voiddispose(List<T>taskList, Consumer<T>consumer) throwsInterruptedException {

       dispose(true, POOL_SIZE, taskList, consumer);

   }

   /**

    * 并行处理,并等待结束

    *

    * @param moreThread 是否多线程执行

    * @param poolSize   线程池大小

    * @param taskList   任务列表

    * @param consumer   消费者

    * @param <T>

    * @throws InterruptedException

    */

   publicstatic<T>voiddispose(booleanmoreThread, intpoolSize, List<T>taskList, Consumer<T>consumer) throwsInterruptedException {

       if (CollectionUtils.isEmpty(taskList)) {

           return;

       }

       //如果是多线程且核心线程数大于一则进入方法

       if (moreThread&&poolSize>1) {

           poolSize=Math.min(poolSize, taskList.size());

           ExecutorServiceexecutorService=null;

           try {

               //新建一个固定大小的线程池 核心线程数为poolSize

               executorService=Executors.newFixedThreadPool(poolSize);

               //juc工具类 用于让必须所有任务都处理完后才进行下一步

               CountDownLatchcountDownLatch=newCountDownLatch(taskList.size());

               for (Titem : taskList) {

                   executorService.execute(() -> {

                       try {

                           //消费任务

                           consumer.accept(item);

                       } finally {

                           //处理完后减一

                           countDownLatch.countDown();

                       }

                   });

               }

               //在此等待 当countDownLatch变成0后才继续进行下一步

               countDownLatch.await();

           } finally {

               if (executorService!=null) {

                   executorService.shutdown();

               }

           }

       } else {

           for (Titem : taskList) {

               consumer.accept(item);

           }

       }

   }

  publicstaticvoidmain(String[] args) throwsInterruptedException {

       //生成1-10的10个数字,放在list中,相当于10个任务

       List<Integer>list=Stream.iterate(1, a->a+1).limit(10).collect(Collectors.toList());

       JSONObjectobject=newJSONObject();

       object.put("name","sss");

       //启动多线程处理list中的数据,每个任务休眠时间为list中的数值

//        Consumer<Integer> c= item -> {

//            try {

//                long startTime = System.currentTimeMillis();

//                object.put("s",item);

//                TimeUnit.SECONDS.sleep(item);

//                long endTime = System.currentTimeMillis();

//                System.out.println(object.toJSONString());

//                System.out.println(System.currentTimeMillis() + ",任务" + item + "执行完毕,耗时:" + (endTime - startTime));

//            } catch (InterruptedException e) {

//                e.printStackTrace();

//            }

//        };

       TaskDisposeUtils.dispose(list, item-> {

           try {

               longstartTime=System.currentTimeMillis();

               object.put("s",item);

               TimeUnit.SECONDS.sleep(item);

               longendTime=System.currentTimeMillis();

               System.out.println(object.toJSONString());

               System.out.println(System.currentTimeMillis() +",任务"+item+"执行完毕,耗时:"+ (endTime-startTime));

           } catch (InterruptedExceptione) {

               e.printStackTrace();

           }

       });

       //上面所有任务处理完毕完毕之后,程序才能继续

       System.out.println(list+"中的任务都处理完毕!");

   }

}

执行结果

🖊️最后总结

🖲要熟练掌握技巧,一定多多坚持练习:骐骥一跃,不能十步;驽马十驾,功在不舍


目录
相关文章
|
3月前
|
JSON 网络协议 安全
【Java】(10)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
232 1
|
3月前
|
JSON 网络协议 安全
【Java基础】(1)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
246 1
|
4月前
|
Java 开发者
Java并发编程:CountDownLatch实战解析
Java并发编程:CountDownLatch实战解析
493 100
|
4月前
|
数据采集 存储 弹性计算
高并发Java爬虫的瓶颈分析与动态线程优化方案
高并发Java爬虫的瓶颈分析与动态线程优化方案
Java 数据库 Spring
200 0
|
存储 缓存 Java
JAVA并发编程系列(11)线程池底层原理架构剖析
本文详细解析了Java线程池的核心参数及其意义,包括核心线程数量(corePoolSize)、最大线程数量(maximumPoolSize)、线程空闲时间(keepAliveTime)、任务存储队列(workQueue)、线程工厂(threadFactory)及拒绝策略(handler)。此外,还介绍了四种常见的线程池:可缓存线程池(newCachedThreadPool)、定时调度线程池(newScheduledThreadPool)、单线程池(newSingleThreadExecutor)及固定长度线程池(newFixedThreadPool)。
|
安全 Java 数据库
一天十道Java面试题----第四天(线程池复用的原理------>spring事务的实现方式原理以及隔离级别)
这篇文章是关于Java面试题的笔记,涵盖了线程池复用原理、Spring框架基础、AOP和IOC概念、Bean生命周期和作用域、单例Bean的线程安全性、Spring中使用的设计模式、以及Spring事务的实现方式和隔离级别等知识点。
|
缓存 监控 Java
(十)深入理解Java并发编程之线程池、工作原理、复用原理及源码分析
深入理解Java并发编程之线程池、工作原理、复用原理及源码分析
248 0
|
监控 Java 开发者
深入理解Java并发编程:线程池的原理与实践
【5月更文挑战第85天】 在现代Java应用开发中,高效地处理并发任务是提升性能和响应能力的关键。线程池作为一种管理线程的机制,其合理使用能够显著减少资源消耗并优化系统吞吐量。本文将详细探讨线程池的核心原理,包括其内部工作机制、优势以及如何在Java中正确实现和使用线程池。通过理论分析和实例演示,我们将揭示线程池对提升Java应用性能的重要性,并给出实践中的最佳策略。
|
缓存 Java
Java面试题:描述Java中的线程池及其实现方式,详细说明其原理
Java面试题:描述Java中的线程池及其实现方式,详细说明其原理
159 0