阿里一面:那我把线程池coreSize配置成0会怎样?

写在前面

设想一下,在我们的项目中存在一个边缘的业务流程,它并不频繁地被触发。在设计线程池时,我回想起了线程池的常见配置原则。为了最大限度地节省资源,我将核心线程数(corePoolSize)设置为0。这样的配置是否能够顺利执行任务呢?

线程池配置原则回顾

在任务提交时,线程池会根据以下策略进行处理:

  • 如果线程池中的线程数少于核心线程数,则创建一个新线程来执行任务。
  • 如果线程池中的线程数大于等于核心线程数:
  • 如果阻塞队列未满,则将任务放入阻塞队列中。
  • 如果阻塞队列已满:
    • 如果线程池中的线程数小于最大线程数,则创建一个新线程来执行当前任务。
    • 如果线程池中的线程数大于等于最大线程数,则执行拒绝策略。

基于上述流程,当corePoolSize设置为0时,任务是否会直接进入阻塞队列等待执行?如果队列足够大,任务是否可能因为队列未满而永远无法执行(因为没有空闲线程被创建)?

实践出真知

猜测无益,实践才是检验真理的唯一标准。下面我配置了一个线程池,希望在提交任务时,能够通过“空闲线程”快速处理任务,然后自动销毁,不占用系统资源。那么,这种配置的线程池是否可能因为阻塞队列未满而导致任务迟迟不被执行?

private static final ThreadPoolExecutor threadPoolExecutor = new ThreadPoolExecutor(  
    0,  
    8,  
    5, TimeUnit.MINUTES,  
    new LinkedBlockingQueue<>(512),  
    new CallerRunsPolicy()  
);

我们使用这个线程池来打印一个字符串,以验证任务是否能够被执行。

public static void main(String[] args) {  
    MonitorThreadPoolConfig.addThreadPoolExecutor(threadPoolExecutor);  

    threadPoolExecutor.execute(() -> System.out.println("hahahhaha "));  
}

执行结果显示:任务确实被执行了(这里我添加了对线程池的定时监控)。

执行结果

深入源码

public void execute(Runnable command) {  
    if (command == null)  
        throw new NullPointerException();  
    /*  
    * 执行分为三个步骤:  
    *  
    * 1. 如果运行的线程数少于核心线程数,尝试  
    * 用给定的命令作为其第一个任务启动一个新线程。  
    * addWorker原子检查runState和workerCount,  
    * 因此可以防止在不应该添加线程时添加线程。  
    *  
    * 2. 如果任务可以成功入队,那么我们仍然需要  
    * 再次检查是否应该添加一个线程(因为现有的线程可能已经死亡)  
    * 或者池是否已经关闭。因此,如果停止了,我们需要回滚入队操作,  
    * 或者如果没有线程,就启动一个新线程。  
    *  
    * 3. 如果我们不能将任务入队,那么我们尝试添加一个新线程。  
    * 如果失败,我们知道我们已经关闭或者饱和了,  
    * 因此拒绝任务。  
    */  
    int c = ctl.get();  
    if (workerCountOf(c) < corePoolSize) {  
        if (addWorker(command, true))  
            return;  
        c = ctl.get();  
        if (isRunning(c) && workQueue.offer(command)) {  
            int recheck = ctl.get();  
            if (! isRunning(recheck) && remove(command))  
                reject(command);  
            else if (workerCountOf(recheck) == 0)  
                addWorker(null, false);  
        }  
        else if (!addWorker(command, false))  
            reject(command);  
    }
}

从源码的第34行到第35行可以看出:当核心线程数为0时,会创建一个新的线程来执行当前任务。

进一步探讨

既然可以通过设置corePoolSize为0来最小化“常驻”线程的资源占用,那么是否有其他方法可以达到相同的效果呢?

有经验的开发者可能会立刻回答:从Java 1.6开始,ThreadPoolExecutor有一个属性可以实现核心线程超时销毁的功能。

/**  
* 如果设置为false(默认),即使空闲,核心线程也会保持活动状态。  
* 如果设置为true,核心线程将使用keepAliveTime来超时等待工作。  
*/  
private volatile boolean allowCoreThreadTimeOut;

我们可以将线程池配置如下

文章整理自互联网,只做测试使用。发布者:Lomu,转转请注明出处:https://www.it1024doc.com/4405.html

(0)
LomuLomu
上一篇 2024 年 12 月 26 日 上午4:21
下一篇 2024 年 12 月 26 日 上午5:22

相关推荐

  • 【java-数据结构篇】揭秘 Java LinkedList:链表数据结构的 Java 实现原理与核心概念

    我的个人主页我的专栏:Java-数据结构 ,希望能帮助到大家!!!点赞❤ 收藏❤ 目录 1. Java LinkedList 基础 1.1 LinkedList 简介1.2 LinkedList 的实现原理1.3 LinkedList 与 ArrayList 的区别 2. 链表基础 2.1 链表的定义与种类2.2 单链表与双链表的区别2.3 循环链表与普通链…

    2024 年 12 月 28 日
    29400
  • 50个JAVA常见代码大全:学完这篇从Java小白到架构师

    50个JAVA常见代码大全:学完这篇从Java小白到架构师 Java,作为一门流行多年的编程语言,始终占据着软件开发领域的重要位置。无论是初学者还是经验丰富的程序员,掌握Java中常见的代码和概念都是至关重要的。本文将列出50个Java常用代码示例,并提供相应解释,助力你从Java小白成长为架构师。 基础语法 1. Hello World “`java p…

    未分类 2025 年 1 月 14 日
    39400
  • Java技术新视野——Java实时大数据处理赋能车联网协同驾驶的实践探索(197)

    ✨亲爱的技术爱好者们,诚挚欢迎访问【云端科技驿站】!在这个数字化浪潮奔涌的时代,我们致力于打造一个融合创新技术与深度思考的知识分享平台。这里不仅有前沿的技术解析,更期待您带来独到见解,让我们携手在科技海洋中扬帆远航!✨全网平台统一标识:云端科技驿站一、加入【技术精英圈】快速通道1:【云端技术交流圈】快速通道2:【CSDN技术创作营】二、核心专栏推荐:1. 【…

    2025 年 5 月 13 日
    11400
  • Nginx HttpHeader增加几个关键的安全选项

    针对像德勤这样的专业渗透测试(Pentest)的场景中,为了确保网站的安全性并通过严格的安全审查,需要为这些安全头配置更细致、专业的参数。 以下是对每个选项的建议以及设置值的详细说明: 1. Strict-Transport-Security (HSTS) 确保所有通信强制通过 HTTPS 并防止降级攻击。 推荐值: add_header Strict-Tr…

    未分类 2024 年 12 月 30 日
    28700
  • IntelliJ IDEA 2024激活破解详细教程(IDEA永久激活码)

    IntelliJ IDEA 是一款广受欢迎的 Java 集成开发环境,以其智能代码助手、代码自动提示和强大的重构功能闻名。然而,对于一些开发者来说,其高昂的订阅费用可能是个不小的负担。本文将介绍一种通过脚本激活 IDEA 的方法,帮助大家免费使用这款强大的开发工具。 一、准备工作 1. 下载并安装最新版本的 IDEA 首先,前往 JetBrains 官网下载…

    未分类 2024 年 6 月 23 日
    1.9K00

发表回复

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

联系我们

400-800-8888

在线咨询: QQ交谈

邮件:admin@example.com

工作时间:周一至周五,9:30-18:30,节假日休息

关注微信