浅谈Java ThreadPoolExecutor的使用

一、前言

线程池主要由以下4个核心组件组成。

  • 线程池管理器:用于创建并管理线程池
  • 工作线程:线程池中执行具体任务的线程
  • 任务接口:用于定义工作线程的调度和执行策略,只有线程实现了该接口,线程中的任务才能被线程池调度
  • 任务队列:放待处理的任务,新的任务将会不断被加入队列中,执行完成的任务将从队列中移除

二、ThreadPoolExecutor

如下是线程池的构造方法

public ThreadPoolExecutor(int corePoolSize,
                              int maximumPoolSize,
                              long keepAliveTime,
                              TimeUnit unit,
                              BlockingQueue<Runnable> workQueue,
                              ThreadFactory threadFactory,
                              RejectedExecutionHandler handler) {
        if (corePoolSize < 0 ||
            maximumPoolSize <= 0 ||
            maximumPoolSize < corePoolSize ||
            keepAliveTime < 0)
            throw new IllegalArgumentException();
        if (workQueue == null || threadFactory == null || handler == null)
            throw new NullPointerException();
        this.acc = System.getSecurityManager() == null ?
                null :
                AccessController.getContext();
        this.corePoolSize = corePoolSize;
        this.maximumPoolSize = maximumPoolSize;
        this.workQueue = workQueue;
        this.keepAliveTime = unit.toNanos(keepAliveTime);
        this.threadFactory = threadFactory;
        this.handler = handler;
}

其中具体参数含义为:

1.corePoolSize:线程池中核心线程的数量

2.maximumPoolSize:线程池中最大线程的数量

3.keepAliveTime:当线程数量超过corePoolSize时,空闲线程的存活时间

4.unit:keepAliveTime的时间单位

5.workQueue:任务队列,被提交但尚未被执行的任务存放的地方

6.threadFactory:线程工厂,用于创建线程,可使用默认的线程工厂或自定义线程工厂

7.handler:由于任务过多或其他原因导致线程池无法处理时的任务拒绝策略

三、构造函数参数解析

编写测试类如下:

public class ThreadPoolSerialTest {
    public static void main(String[] args) {
        //核心线程数
        int corePoolSize = 2;
        //最大线程数
        int maximumPoolSize = 4;
        //超过corePoolSize线程数量的线程最大空闲时间
        long keepAliveTime = 2;
        //以秒为时间单位
        TimeUnit unit = TimeUnit.SECONDS;
        //创建工作队列,用于存放提交的等待执行任务
        BlockingQueue<Runnable> workQueue = new ArrayBlockingQueue<>(2);
        ThreadPoolExecutor threadPoolExecutor = null;

        try {
            // 1.创建线程池
            threadPoolExecutor = new ThreadPoolExecutor(corePoolSize,
                    maximumPoolSize,
                    keepAliveTime,
                    unit,
                    workQueue,
                    new ThreadPoolExecutor.AbortPolicy());
            // 2.循环提交任务
            for (int i = 0; i < 6; i++) {
                //提交任务的索引
                final int index = (i+1);
                threadPoolExecutor.submit(()->{
                    //线程打印输出
                    System.out.println("大家好,我是线程:"+index);
                    try {
                        //模拟线程执行时间,10s
                        Thread.sleep(10000);
                        System.out.println("线程:"+index+"运行完毕");
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                });
                //每个任务提交后休眠500ms再提交下一个任务,用于保证提交顺序
                Thread.sleep(500);
            }
        } catch (InterruptedException e) {
            e.printStackTrace();
        } finally {
            // 3.关闭线程池
            threadPoolExecutor.shutdown();
        }
    }
}

其中循环了6次,让线程池执行了6次任务,恰好满足maximumPoolSize+workQueue容量=并发执行任务数。输出结果如下:

大家好,我是线程:1
大家好,我是线程:2
大家好,我是线程:5
大家好,我是线程:6
线程:1运行完毕
大家好,我是线程:3
线程:2运行完毕
大家好,我是线程:4
线程:5运行完毕
线程:6运行完毕
线程:3运行完毕
线程:4运行完毕

这段输出看似没有规律,其实这里输出完全是由线程池控制的;下面就来分行解析输出:

大家好,我是线程:1
大家好,我是线程:2
大家好,我是线程:5
大家好,我是线程:6

1.全新线程池被创建后,有Runnable或CallBack接口的实现被提交给线程池执行;线程池的corePoolSize=2,此时前两个任务提交后就立即执行,便输出了线程1 线程2

2.此时仍继续向线程池提交任务,线程池中workQueue容量=2,被加入的任务存放到任务队列中,即把线程3 线程4存放到了任务队列中;

3.任务队列充满后,仍继续向线程池提交任务,线程池的maximumPoolSize=4,除开核心线程数2个外还允许创建4-2个线程来执行任务,便输出了线程5 线程6

线程:1运行完毕
大家好,我是线程:3
线程:2运行完毕
大家好,我是线程:4

1.线程:1运行完毕:表示第一个线程任务执行完毕了

2.大家好,我是线程:3:线程1运行完毕后,此时线程池中有一个空闲的线程,第一个进入任务队列中的任务第一个交给线程处理

3.线程:2运行完毕 大家好,我是线程:4 :和上面线程执行完毕,任务对列中任务执行一致

线程:5运行完毕
线程:6运行完毕
线程:3运行完毕
线程:4运行完毕

因为每一个任务的执行时间控制的是一样的,此时输出的内容便是先被线程池执行的任务先执行完毕。

四、总结

线程池刚被创建时,只是向系统申请一个用于执行线程队列和管理线程池的资源。在调用execute()添加一个任务时,线程池会按照以下流程执行任务:

正在运行的线程数量a:a<corePoolSize,线程池立即创建线程并执行任务;若此时a=corePoolSize,则任务被存放到workQueue任务队列中,直到任务队列被充满

任务队列workQueue已充满且正在运行的线程数a:a<maximumPoolSize,线程池立即创建非核心线程并执行任务;若有任务执行完毕,该任务将被线程池队列中移除,线程池从队列中取先入队的任务执行;当线程处于空闲状态的时间超过keepAliveTime时间时,正在运行的线程数acorePoolSize<a,线程池停止空闲的线程。线程池将任务执行完毕后,线程池会收缩到corePoolSize大小

任务队列workQueue已充满且正在运行的线程数a:a=maximumPoolSize,线程池拒绝执行该任务并抛出RejectExecutionException异常

到此这篇关于浅谈Java ThreadPoolExecutor的使用的文章就介绍到这了,更多相关Java ThreadPoolExecutor内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • Java ThreadPoolExecutor 线程池的使用介绍

    Executors Executors 是一个Java中的工具类. 提供工厂方法来创建不同类型的线程池. 从上图中也可以看出, Executors的创建线程池的方法, 创建出来的线程池都实现了 ExecutorService接口. 常用方法有以下几个: newFixedThreadPool(int Threads): 创建固定数目线程的线程池, 超出的线程会在队列中等待. newCachedThreadPool(): 创建一个可缓存线程池, 如果线程池长度超过处理需要, 可灵活回收空闲线程(60

  • java多线程CountDownLatch与线程池ThreadPoolExecutor/ExecutorService案例

    1.CountDownLatch: 一个同步工具类,它允许一个或多个线程一直等待,直到其他线程的操作执行完后再执行. 2.ThreadPoolExecutor/ExecutorService: 线程池,使用线程池可以复用线程,降低频繁创建线程造成的性能消耗,同时对线程的创建.启动.停止.销毁等操作更简便. 3.使用场景举例: 年末公司组织团建,要求每一位员工周六上午8点到公司门口集合,统一乘坐公司所租大巴前往目的地. 在这个案例中,公司作为主线程,员工作为子线程. 4.代码示例: package

  • 浅谈Java线程池的7大核心参数

    前言 java中经常需要用到多线程来处理一些业务,我不建议单纯使用继承Thread或者实现Runnable接口的方式来创建线程,那样势必有创建及销毁线程耗费资源.线程上下文切换问题. 同时创建过多的线程也可能引发资源耗尽的风险,这个时候引入线程池比较合理,方便线程任务的管理. java中涉及到线程池的相关类均在jdk1.5开始的java.util.concurrent包中,涉及到的几个核心类及接口包括: Executor.Executors.ExecutorService.ThreadPoolE

  • java 定时器线程池(ScheduledThreadPoolExecutor)的实现

    前言 定时器线程池提供了定时执行任务的能力,即可以延迟执行,可以周期性执行.但定时器线程池也还是线程池,最底层实现还是ThreadPoolExecutor,可以参考我的另外一篇文章多线程–精通ThreadPoolExecutor. 特点说明 1.构造函数 public ScheduledThreadPoolExecutor(int corePoolSize) { // 对于其他几个参数在ThreadPoolExecutor中都已经详细分析过了,所以这里,将不再展开 // 这里我们可以看到调用基类

  • 深入理解Java线程池从设计思想到源码解读

    线程池:从设计思想到源码解析 前言初识线程池线程池优势线程池设计思路 深入线程池构造方法任务队列拒绝策略线程池状态初始化&容量调整&关闭 使用线程池ThreadPoolExecutorExecutors封装线程池 解读线程池execute()addWorker()Worker类runWorker()processWorkerExit() 前言 各位小伙伴儿,春节已经结束了,在此献上一篇肝了一个春节假期的迟来的拜年之作,希望读者朋友们都能有收获. 根据穆氏哲学,投入越多,收获越大.我作此文时

  • Java多线程之线程池七个参数详解

    ThreadPoolExecutor是JDK中的线程池实现,这个类实现了一个线程池需要的各个方法,它提供了任务提交.线程管理.监控等方法. 下面是ThreadPoolExecutor类的构造方法源码,其他创建线程池的方法最终都会导向这个构造方法,共有7个参数:corePoolSize.maximumPoolSize.keepAliveTime.unit.workQueue.threadFactory.handler. public ThreadPoolExecutor(int corePoolS

  • Java 自定义线程池和线程总数控制操作

    1 概述 池化是常见的思想,线程池是非常典型的池化的实现,<Java并发编程实战>也大篇幅去讲解了Java中的线程池.本文实现一个简单的线程池. 2 核心类 [1]接口定义 public interface IThreadPool<Job extends Runnable> { /** * 关闭线程池 */ public void shutAlldown(); /** * 执行任务 * * @param job 任务 */ public void execute(Job job);

  • java线程池对象ThreadPoolExecutor的深入讲解

    使用线程池的好处 1.降低资源消耗 可以重复利用已创建的线程降低线程创建和销毁造成的消耗. 2.提高响应速度 当任务到达时,任务可以不需要等到线程创建就能立即执行. 3.提高线程的可管理性 线程是稀缺资源,如果无限制地创建,不仅会消耗系统资源,还会降低系统的稳定性,使用线程池可以进行统一分配.调优和监控 ThreadPoolExecutor 介绍: java 提供的线程池类: ThreadPoolExecutor 作用: 两个作用: 1,用于分离执行任务和当前线程: 2,主要设计初衷:重复利用T

  • Java利用线程工厂监控线程池的实现示例

    ThreadFactory 线程池中的线程从哪里来呢?就是ThreadFoctory public interface ThreadFactory { Thread newThread(Runnable r); } Threadfactory里面有个接口,当线程池中需要创建线程就会调用该方法,也可以自定义线程工厂 public class ThreadfactoryText { public static void main(String[] args) { Runnable runnable=

  • Java简单实现线程池

    本文实例为大家分享了Java简单实现线程池的具体代码,供大家参考,具体内容如下 一.线程池 线程池是一种缓冲提高效率的技术. 相当于一个池子,里面存放大量已经创建好的线程,当有一个任务需要处理时, 可以直接从池子里面取一个线程去执行它. 包括内存池,很多缓冲的技术都是采用这种技术. 其实理解起来很简答! 为什么需要线程池,这种池的技术? 1.1 减少开辟资源和销毁资源带来的损耗. 开辟线程,申请内存(具体的可以看C语言中malloc底层实现原理),销毁线程.释放内存资源等一些操作都是有时间消耗的

  • Java 使用线程池执行多个任务的示例

    在执行一系列带有IO操作(例如下载文件),且互不相关的异步任务时,采用多线程可以很极大的提高运行效率.线程池包含了一系列的线程,并且可以管理这些线程.例如:创建线程,销毁线程等.本文将介绍如何使用Java中的线程池执行任务. 1 任务类型 在使用线程池执行任务之前,我们弄清楚什么任务可以被线程池调用.按照任务是否有返回值可以将任务分为两种,分别是实现Runnable的任务类(无参数无返回值)和实现Callable接口的任务类(无参数有返回值).在打代码时根据需求选择对应的任务类型. 1.1 实现

  • Java线程池ThreadPoolExecutor原理及使用实例

    引导 要求:线程资源必须通过线程池提供,不允许在应用自行显式创建线程: 说明:使用线程池的好处是减少在创建和销毁线程上所花的时间以及系统资源的开销,解决资源不足的问题.如果不使用线程池,有可能造成系统创建大量同类线程而导致消耗内存或者"过度切换"的问题. 线程池介绍线程池概述   线程池,顾名思义是一个放着线程的池子,这个池子的线程主要是用来执行任务的.当用户提交任务时,线程池会创建线程去执行任务,若任务超过了核心线程数的时候,会在一个任务队列里进行排队等待,这个详细流程,我们会后面细

随机推荐