Java线程池七个参数详解:核心线程数、最大线程数、空闲线程存活时间、时间单位、工作队列、线程工厂、拒绝策略

Java线程池七个参数详解:核心线程数、最大线程数、空闲线程存活时间、时间单位、工作队列、线程工厂、拒绝策略源码简介 ThreadPoolEx 是 JDK 中的线程池实现 这个类实现了一个线程池需要的各个方法 它实现了任务提交 线程管理 监控等等方法 来看看 ThreadPoolEx 类的构造方法源码 其他创建线程池的方法最终都会导向这个构造方法 共有 7 个参数 corePoolSize maximumPoolS keepAliveTim unit workQueue threadFactor handler 下面将对这 7 个参数展开讲解 publicThread

源码简介

ThreadPoolExecutor是JDK中的线程池实现,这个类实现了一个线程池需要的各个方法,它提供了任务提交、线程管理、监控等方法。

下面是ThreadPoolExecutor类的构造方法源码,其他创建线程池的方法最终都会导向这个构造方法,共有7个参数:corePoolSize、maximumPoolSize、keepAliveTime、unit、workQueue、threadFactory、handler。

 public ThreadPoolExecutor(int corePoolSize, int maximumPoolSize, long keepAliveTime, TimeUnit unit, BlockingQueue 
  
    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; } 
  

这些参数都通过volatile修饰:

public class ThreadPoolExecutor extends AbstractExecutorService { private final BlockingQueue 
  
    workQueue; private volatile ThreadFactory threadFactory; private volatile RejectedExecutionHandler handler; private volatile long keepAliveTime; // 是否允许核心线程被回收 private volatile boolean allowCoreThreadTimeOut; private volatile int corePoolSize; private volatile int maximumPoolSize; } 
  

corePoolSize:核心线程数

线程池维护的最小线程数量,核心线程创建后不会被回收(注意:设置allowCoreThreadTimeout=true后,空闲的核心线程超过存活时间也会被回收)。

大于核心线程数的线程,在空闲时间超过keepAliveTime后会被回收。

线程池刚创建时,里面没有一个线程,当调用 execute() 方法添加一个任务时,如果正在运行的线程数量小于corePoolSize,则马上创建新线程并运行这个任务。

maximumPoolSize:最大线程数

线程池允许创建的最大线程数量。

当添加一个任务时,核心线程数已满,线程池还没达到最大线程数,并且没有空闲线程,工作队列已满的情况下,创建一个新线程并执行。

keepAliveTime:空闲线程存活时间

当一个可被回收的线程的空闲时间大于keepAliveTime,就会被回收。

可被回收的线程:

  1. 设置allowCoreThreadTimeout=true的核心线程。
  2. 大于核心线程数的线程(非核心线程)。

unit:时间单位

keepAliveTime的时间单位:

TimeUnit.NANOSECONDS TimeUnit.MICROSECONDS TimeUnit.MILLISECONDS // 毫秒 TimeUnit.SECONDS TimeUnit.MINUTES TimeUnit.HOURS TimeUnit.DAYS

workQueue:工作队列

存放待执行任务的队列:当提交的任务数超过核心线程数大小后,再提交的任务就存放在工作队列,任务调度时再从队列中取出任务。它仅仅用来存放被execute()方法提交的Runnable任务。工作队列实现了BlockingQueue接口。

JDK默认的工作队列有五种:

  1. ArrayBlockingQueue 数组型阻塞队列:数组结构,初始化时传入大小,有界,FIFO,使用一个重入锁,默认使用非公平锁,入队和出队共用一个锁,互斥。
  2. LinkedBlockingQueue 链表型阻塞队列:链表结构,默认初始化大小为Integer.MAX_VALUE,有界(近似无解),FIFO,使用两个重入锁分别控制元素的入队和出队,用Condition进行线程间的唤醒和等待。
  3. SynchronousQueue 同步队列:容量为0,添加任务必须等待取出任务,这个队列相当于通道,不存储元素。
  4. PriorityBlockingQueue 优先阻塞队列:无界,默认采用元素自然顺序升序排列。
  5. DelayQueue 延时队列:无界,元素有过期时间,过期的元素才能被取出。

threadFactory:线程工厂

创建线程的工厂,可以设定线程名、线程编号等。

默认线程工厂:

 / * The default thread factory */ static class DefaultThreadFactory implements ThreadFactory { private static final AtomicInteger poolNumber = new AtomicInteger(1); private final ThreadGroup group; private final AtomicInteger threadNumber = new AtomicInteger(1); private final String namePrefix; DefaultThreadFactory() { SecurityManager s = System.getSecurityManager(); group = (s != null) ? s.getThreadGroup() : Thread.currentThread().getThreadGroup(); namePrefix = "pool-" + poolNumber.getAndIncrement() + "-thread-"; } public Thread newThread(Runnable r) { Thread t = new Thread(group, r, namePrefix + threadNumber.getAndIncrement(), 0); if (t.isDaemon()) t.setDaemon(false); if (t.getPriority() != Thread.NORM_PRIORITY) t.setPriority(Thread.NORM_PRIORITY); return t; } }

handler:拒绝策略

当线程池线程数已满,并且工作队列达到限制,新提交的任务使用拒绝策略处理。可以自定义拒绝策略,拒绝策略需要实现RejectedExecutionHandler接口。

JDK默认的拒绝策略有四种:

  1. AbortPolicy:丢弃任务并抛出RejectedExecutionException异常。
  2. DiscardPolicy:丢弃任务,但是不抛出异常。可能导致无法发现系统的异常状态。
  3. DiscardOldestPolicy:丢弃队列最前面的任务,然后重新提交被拒绝的任务。
  4. CallerRunsPolicy:由调用线程处理该任务。

默认拒绝策略:

 / * The default rejected execution handler */ private static final RejectedExecutionHandler defaultHandler = new AbortPolicy(); public static class AbortPolicy implements RejectedExecutionHandler { / * Creates an {@code AbortPolicy}. */ public AbortPolicy() { } / * Always throws RejectedExecutionException. * * @param r the runnable task requested to be executed * @param e the executor attempting to execute this task * @throws RejectedExecutionException always */ public void rejectedExecution(Runnable r, ThreadPoolExecutor e) { throw new RejectedExecutionException("Task " + r.toString() + " rejected from " + e.toString()); } }

线程池的执行流程

Java线程池七个参数详解:核心线程数、最大线程数、空闲线程存活时间、时间单位、工作队列、线程工厂、拒绝策略

自定义线程池工具

import java.util.concurrent.*; import java.util.concurrent.atomic.AtomicInteger; / * 线程池工厂工具 * * @author 向振华 * @date 2021/04/11 10:24 */ public class ThreadPoolFactory { / * 生成固定大小的线程池 * * @param threadName 线程名称 * @return 线程池 */ public static ExecutorService createFixedThreadPool(String threadName) { AtomicInteger threadNumber = new AtomicInteger(0); return new ThreadPoolExecutor( // 核心线程数 desiredThreadNum(), // 最大线程数 desiredThreadNum(), // 空闲线程存活时间 60L, // 空闲线程存活时间单位 TimeUnit.SECONDS, // 工作队列 new ArrayBlockingQueue<>(1024), // 线程工厂 new ThreadFactory() { @Override public Thread newThread(Runnable r) { return new Thread(r, threadName + "-" + threadNumber.getAndIncrement()); } }, // 拒绝策略 new RejectedExecutionHandler() { @Override public void rejectedExecution(Runnable r, ThreadPoolExecutor executor) { if (!executor.isShutdown()) { try { //尝试阻塞式加入任务队列 executor.getQueue().put(r); } catch (Exception e) { //保持线程的中断状态 Thread.currentThread().interrupt(); } } } }); } / * 理想的线程数,使用 2倍cpu核心数 */ public static int desiredThreadNum() { return Runtime.getRuntime().availableProcessors() * 2; } }

 

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/216804.html原文链接:https://javaforall.net

(0)
上一篇 2026年3月18日 上午11:01
下一篇 2026年3月18日 上午11:02


相关推荐

  • 快速入门Win+R命令(附图)

    快速入门Win+R命令(附图)最近在学习 Linux 被命令行深深吸引了 陷入其中不能自拔 考虑到 Windows 上也有 cmd 命令行 但对新人来说不是很友好 这次我们就先讲一下 Win R 运行框里的快捷键 绝对能提高不少效率 什么是 Win R 防止有些小白看不懂 所以说明一些 使用 Windows R 快捷键就可以打开如下图的运行窗口 在里面输入命令可以方便快捷地打开很多东西 而且本文的所有操作都是在这个运行框里输入的 不要与 cm

    2026年3月17日
    2
  • loadrunner12安装教程_word入门基础教程视频

    loadrunner12安装教程_word入门基础教程视频Loadrunner安装_简单使用基础教程,包括VirtualUserGenerator、Controller、Analysis的简单使用以及对Loadrunner的介绍。

    2022年10月14日
    7
  • 史上最简单的 GitHub 教程

    史上最简单的 GitHub 教程1简介  GitHub是一个面向开源及私有软件项目的托管平台,因为只支持Git作为唯一的版本库格式进行托管,故名GitHub。GitHub于2008年4月10日正式上线,除了Git代码仓库托管及基本的Web管理界面以外,还提供了订阅、讨论组、文本渲染、在线文件编辑器、协作图谱(报表)、代码片段分享(Gist)等功能。目前,在GitHub上托管的版本数量非常之多,其

    2022年7月22日
    11
  • 小白也能养虾 联想免费装OpenClaw:线上一键+线下安装

    小白也能养虾 联想免费装OpenClaw:线上一键+线下安装

    2026年3月14日
    4
  • C++ STL它vector详细解释

    C++ STL它vector详细解释

    2022年1月8日
    52
  • python redis pubsub_redis之PubSub

    python redis pubsub_redis之PubSub前面我们讲了 Redis 消息队列的使用方法 但是没有提到 Redis 消息队列的不足之处 那就是它不支持消息的多播机制 消息多播消息多播允许生产者生产一次消息 中间件负责将消息复制到多个消息队列 每个消息队列由相应的消费组进行消费 它是分布式系统常用的一种解耦方式 用于将多个消费组的逻辑进行拆分 支持了消息多播 多个消费组的逻辑就可以放到不同的子系统中 如果是普通的消息队列 就得将多个不同的消费

    2026年3月18日
    2

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号