ICode9

精准搜索请尝试: 精确搜索
首页 > 其他分享> 文章详细

Executors创建线程池原理

2021-12-13 16:31:47  阅读:223  来源: 互联网

标签:Executors 队列 创建 任务 线程 核心 new


Executors创建线程池原理

1、ExecutorService接口

  • 通过构造参数创建
  • 通过Executors创建

实现类ThreadPoolExecutor构造参数为:

image-20210421141202421

//全参构造函数
public ThreadPoolExecutor(int corePoolSize,
                              int maximumPoolSize,
                              long keepAliveTime,
                              TimeUnit unit,
                              BlockingQueue<Runnable> workQueue,
                              ThreadFactory threadFactory,
                              RejectedExecutionHandler handler) {
  • corePoolSize:核心线程数
  • maximumPoolSize:最大线程数
  • keepAliveTime:超过核心线程数的线程空闲时间
  • unit:空闲时间的单位
  • workQueue:工作队列
  • threadFactory:创建线程的工厂
  • handler:任务超出队列后的处理策略

2、线程池的工作流程

(1)执行新的任务,如果线程数还没有超过核心线程数,就创建线程执行任务

(2)如果线程数已经超过核心线程数,就将任务放入工作队列

(3)如果线程数已经超过核心线程数且工作队列已满,查看线程数是否超过最大线程数,如果没有,就创建新的线程,将工作队列中的任务放入线程中执行。

(4)如果达到最大线程数,就将新的任务交给处理策略进行处理

(5)当核心线程数可以处理所有的任务时,超过核心线程数的线程空闲时间超过keepAliveTime后,就销毁线程。

image-20210421142514292

3、Executors常用的线程池

(1)newFixedThreadPool

image-20210421143049603

单参数,指的是线程数,指定核心线程数和最大线程数,都通过该参数指定

 public static ExecutorService newFixedThreadPool(int nThreads) {
        return new ThreadPoolExecutor(nThreads, nThreads,
                                      0L, TimeUnit.MILLISECONDS,
                                      new LinkedBlockingQueue<Runnable>());
    }
  • 核心线程数和最大线程数使用同样的数量
  • 超过核心线程数的线程空闲时间设置为0(不会超过)
  • 工作队列使用的是LinkedBlockingQueue,无界阻塞队列,该队列可以无限制的放入任务

(2)newSingleThreadExecutor

默认无参,也可传入ThreadFactory自定义创建线程的工厂

public static ExecutorService newSingleThreadExecutor() {
        return new FinalizableDelegatedExecutorService
            (new ThreadPoolExecutor(1, 1,
                                    0L, TimeUnit.MILLISECONDS,
                                    new LinkedBlockingQueue<Runnable>()));
    }
  • 创建核心线程数是一个线程的线程池
  • 超时时间为0
  • 无界阻塞队列

(3)newCachedThreadPool

默认无参,也可传入ThreadFactory自定义创建线程的工厂

return new ThreadPoolExecutor(0, Integer.MAX_VALUE,
                                      60L, TimeUnit.SECONDS,
                                      new SynchronousQueue<Runnable>());
  • 创建核心线程数为0,最大线程数为Integer最大值
  • 当池中的线程空闲时间超过60秒,销毁线程
  • SynchronousQueue工作队列,属于同步移交队列,不是一个真正的队列,要放入这个队列,必须要有线程等待接收任务,才会放入队列,并移交给线程

(4)newScheduledThreadPool

image-20210421145203543

单个参数传入核心线程数

public static ScheduledExecutorService newScheduledThreadPool(int corePoolSize) {
        return new ScheduledThreadPoolExecutor(corePoolSize);

该线程池,支持定时以及周期性的执行任务。

ScheduledThreadPoolExecutor构造参数:

image-20210421145434679

传入父类的参数中:核心线程数为参数,最大线程数为Integer最大值,超过核心线程数的线程空闲时间0,阻塞队列使用DelayedWorkQueue延迟工作队列。

其父类中使用如下参数,除去传递的参数,Factory使用默认的线程工厂,饱和策略使用默认的策略defaultHandler=AbortPolicy,当任务超出队列后抛出异常。

public ThreadPoolExecutor(int corePoolSize,
                              int maximumPoolSize,
                              long keepAliveTime,
                              TimeUnit unit,
                              BlockingQueue<Runnable> workQueue) {
        this(corePoolSize, maximumPoolSize, keepAliveTime, unit, workQueue,
             Executors.defaultThreadFactory(), defaultHandler);
    }

4、饱和策略

当核心线程数已满,任务队列已满且达到了最大线程数时。

(1)AbortPolicy:抛出异常,让使用者自己捕获异常进行处理

image-20210421151837226

(2)DiscardPolicy:什么也不做,不抛异常

image-20210421151912220

(3)DiscardOldestPolicy:丢弃队列中队首的任务,处理新的任务,这种情况下,如果队列使用的时优先级队列,会抛出优先级最高的任务,不建议将该策略与优先级队列一同使用。

image-20210421151941417

(4)CallerRunsPolicy: 既不抛弃任务也不抛出异常,直接运行任务的run方法,将任务回退给调用者来直接运行。使用该策略时线程池饱和后将由调用线程池的主线程自己来执行任务,因此在执行任务的这段时间里主线程无法再提交新任务,从而使线程池中工作线程有时间将正在处理的任务处理完成。

image-20210421152022620

5、任务队列类型

  • 无界阻塞队列:没有大小限制的队列,常用的无界队列为 LinkedBlockingQueue ,当某个任务执行比较耗时时,新的任务都会堆积在该队列中

  • 有界阻塞队列:

    常用的有两类,一类是遵循FIFO原则的队列如ArrayBlockingQueue与有界的LinkedBlockingQueue,另一类是优先级队列如PriorityBlockingQueue。PriorityBlockingQueue中的优先级由任务的Comparator决定。
    使用有界队列时队列大小需和线程池大小互相配合,线程池较小有界队列较大时可减少内存消耗,降低cpu使用率和上下文切换,但是可能会限制系统吞吐量。

  • 同步移交队列: 如果不希望任务在队列中等待而是希望将任务直接移交给工作线程,可使用SynchronousQueue作为等待队列。SynchronousQueue不是一个真正的队列,而是一种线程之间移交的机制。要将一个元素放入SynchronousQueue中,必须有另一个线程正在等待接收这个元素。只有在使用无界线程池或者有饱和策略时才建议使用该队列。

image-20210421152710414

原文地址: https://www.pianshen.com/article/252286830/

标签:Executors,队列,创建,任务,线程,核心,new
来源: https://www.cnblogs.com/YpfBolg/p/15683968.html

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有