面试官:核心线程数为0时,线程池如何执行?

· 浏览次数 : 0

小编点评

**当核心线程数为 0 时,线程池是如何执行的?** 当核心线程数为 0 时,线程池会根据以下步骤执行任务: 1. 判断当前工作线程数是否大于核心线程数。 2. 如果当前工作线程数大于核心线程数,则新建线程并执行任务。 3. 判断任务队列是否已满?如果已满,则把任务添加到任务队列中等待线程执行。 4. 判断最大线程数。如果超过最大线程数,则将执行线程池的拒绝策略。 **线程池的核心线程数和最大线程数的作用:** * **核心线程数:**定义了线程池中最小线程数量,即使这些线程处于空闲状态,也不会被销毁。 * **最大线程数:**定义了线程池中允许的最大线程数量,等于核心线程数 + 临时线程数。最大线程数主要是提供了一种机制来应对突发的高并发请求,当有大量任务的时候,可以创建线程数量的上线。 **线程池执行流程的正确性:** 线程池的执行流程确实比我们预期的那般复杂,但它始终遵循以下基本流程: 1. 判断当前工作线程数是否大于核心线程数。 2. 如果当前工作线程数大于核心线程数,则新建线程并执行任务。 3. 判断任务队列是否已满?如果已满,则把任务添加到任务队列中等待线程执行。 4. 判断最大线程数。如果超过最大线程数,则将执行线程池的拒绝策略。 **结论:** 当核心线程数为 0 时,线程池会根据上述步骤执行任务,并最终按照正常逻辑执行。

正文

线程池是 Java 中用于提升程序执行效率的主要手段,也是并发编程中的核心实现技术,并且它也被广泛的应用在日常项目的开发之中。那问题来了,如果把线程池中的核心线程数设置为 0 时,线程池是如何执行的?

要回答这个问题,我们首先要了解在正常情况下,线程池的执行流程,也就是说当有一个任务来了之后,线程池是如何运行的?

1.线程池的执行流程

正常情况下(核心线程数不为 0 的情况下)线程池的执行流程如下:

  1. 判断核心线程数:先判断当前工作线程数是否大于核心线程数,如果结果为 false,则新建线程并执行任务。
  2. 判断任务队列:如果大于核心线程数,则判断任务队列是否已满?如果结果为 false,则把任务添加到任务队列中等待线程执行。
  3. 判断最大线程数:如果任务队列已满,则判断当前线程数量是否超过最大线程数?如果结果为 false,则新建线程执行此任务。
  4. 判断是否要执行拒绝策略:如果超过最大线程数,则将执行线程池的拒绝策略。

如下图所示:

核心线程数 VS 最大线程数

核心线程数(corePoolSize)和最大线程数(maximumPoolSize)都是线程池中的两个重要参数,其中:

  1. 核心线程数定义了线程池中最小线程数量,即使这些线程处于空闲状态,也不会被销毁。
  2. 最大线程数定义了线程池中允许的最大线程数量,最大线程数等于核心线程数 + 临时线程数,最大线程数主要是提供了一种机制来应对突发的高并发请求,当有大量任务的时候,可以创建线程数量的上线。

PS:在线程池的使用过程中,最大线程数必须大于等于核心线程数,否则程序执行会报错。

2.核心线程为0的执行流程

那么问题来了,按照线程池的正常执行流程来看,如果核心线程数为 0 的话,那么当任务来了之后会判断当前工作的线程数不大于核心线程数,那也就不会创建线程执行任务了,会将任务放到队列。

但这个结果又很滑稽,有任务来了线程池竟然不执行,而是先放到任务队列中,这好像有比较奇怪,这就好比你开了一个快递店,当有快递来了之后,你想的不是如何派送,而是直接把它丢到仓库一样滑稽,这会让等快递的人很着急,所以,我们需要验证一下线程池的执行是否如我们猜想的那般,验证代码如下:

// 线程池核心线程数设置为 0
ThreadPoolExecutor threadPoolExecutor = new ThreadPoolExecutor(
                0, 10, 1, TimeUnit.SECONDS,
                new ArrayBlockingQueue<>(10));
// 给线程池添加任务
threadPoolExecutor.execute(() -> {
    System.out.println("www.javacn.site");
});

最终程序的执行结果如下:
image.png
咦,结果怎么不符合我们的预期呢?为什么任务来了之后,没有将任务放到任务队列而是直接执行了呢?

虽然程序执行结果符合正常逻辑,但又和我们了解的线程池执行理论相违背,这是什么原因呢?

查看线程池的执行源码

带着这个疑问,我们查看了线程池的执行源码发现,线程池的执行过程远比我们想想的复杂,线程池核心源码如下:
image.png
从上面源码可以看出,当我们将任务添加到队列的时候,线程池会判断工作的线程数是否为 0,也就是上面圈出来的那行代码,如果当前工作线程为 0 的话,会创建线程执行任务。哦,原来如此,这样,就能将理论和实践对应上了。

也就是说,当核心线程数为 0 时,当来了一个任务之后,会先将任务添加到任务队列,同时也会判断当前工作的线程数是否为 0,如果为 0,则会创建线程来执行线程池的任务,这就是正确的线程池执行流程,同时也是面试官想要的答案。

课后思考

如何实现线程池的监控?如何动态调整线程池的核心线程数呢?

本文已收录到我的面试小站 www.javacn.site,其中包含的内容有:Redis、JVM、并发、并发、MySQL、Spring、Spring MVC、Spring Boot、Spring Cloud、MyBatis、设计模式、消息队列等模块。

与面试官:核心线程数为0时,线程池如何执行?相似的内容:

面试官:核心线程数为0时,线程池如何执行?

线程池是 Java 中用于提升程序执行效率的主要手段,也是并发编程中的核心实现技术,并且它也被广泛的应用在日常项目的开发之中。那问题来了,如果把线程池中的核心线程数设置为 0 时,线程池是如何执行的? 要回答这个问题,我们首先要了解在正常情况下,线程池的执行流程,也就是说当有一个任务来了之后,线程池

小米面试:如何实现优先级线程池?

我们知道,线程池中的所有线程都是由统一的线程工厂来创建的,当我们指定线程工厂时,线程池中的所有线程会使用我们指定的线程工厂来创建线程;但如果没有指定线程工厂,则会使用默认的线程工厂 DefaultThreadFactory 来创建线程,核心源码如下: DefaultThreadFactory() {

解锁Java面试中的锁:深入了解不同类型的锁和它们的用途

简介 多线程编程在现代软件开发中扮演着至关重要的角色。它使我们能够有效地利用多核处理器和提高应用程序的性能。然而,多线程编程也伴随着一系列挑战,其中最重要的之一就是处理共享资源的线程安全性。在这个领域,锁(Lock)是一个关键的概念,用于协调线程之间对共享资源的访问。本文将深入探讨Java中不同类型

面试官:说说Netty的核心组件?

Netty 核心组件是指 Netty 在执行过程中所涉及到的重要概念,这些核心组件共同组成了 Netty 框架,使 Netty 框架能够正常的运行。 Netty 核心组件包含以下内容: 启动器 Bootstrap/ServerBootstrap 事件循环器 EventLoopGroup/EventL

面试官:Dubbo一次RPC请求经历哪些环节?

大家好,我是三友~~ 今天继续探秘系列,扒一扒一次RPC请求在Dubbo中经历的核心流程。 本文是基于Dubbo3.x版本进行讲解 一个简单的Demo 这里还是老样子,为了保证文章的完整性和连贯性,方便那些没有使用过的小伙伴更加容易接受文章的内容,这里快速讲一讲Dubbo一个简单的Demo 如果你已

面试官:什么是双亲委派模型?

本文已经收录进 JavaGuide(「Java学习+面试指南」一份涵盖大部分 Java 程序员所需要掌握的核心知识。) 参加过校招面试的同学,应该对这个问题不陌生。一般提问 JVM 知识点的时候,就会顺带问你双亲委派模型(别扭的翻译。。。)。 就算是不准备面试,学习双亲委派模型对于我们也非常有帮助。

上周面了百度,问的很细~

上周刚刚面了百度,问的问题不算很难,但却很细,我把这些面试题和答案都整理出来了,一起来看吧。 重点介绍一个你觉得有意义的项目? 回答技巧和思路: 介绍的项目业务难度和技术难点要高一些,最好是微服务项目。 简明扼要的讲清楚项目核心板块的业务场景即可,切忌不要讲的太细和太久,这只是面试官要考察你技术问题

Oracle JDK 和 OpenJDK 有什么区别?

摘自 JavaGuide (「Java学习+面试指南」一份涵盖大部分 Java 程序员所需要掌握的核心知识。准备 Java 面试,首选 JavaGuide!) 可能在看这个问题之前很多人和我一样并没有接触和使用过 OpenJDK 。那么 Oracle JDK 和 OpenJDK 之间是否存在重大差异

Java 自增自减运算符和移位运算符介绍

摘自 JavaGuide (「Java学习+面试指南」一份涵盖大部分 Java 程序员所需要掌握的核心知识。准备 Java 面试,首选 JavaGuide!) 自增自减运算符 在写代码的过程中,常见的一种情况是需要某个整数类型变量增加 1 或减少 1,Java 提供了一种特殊的运算符,用于这种表达式

哈啰面试:说说Dubbo运行原理?

Dubbo 是一款高性能、轻量级的开源 RPC(远程过程调用)框架,主要用于构建分布式服务和微服务架构。那 Dubbo 又是如何运行的呢?让我们一起来看。 1.核心组件 要说 Dubbo 运行流程就不得不先来了解一下 Dubbo 的核心组件了,因为 Dubbo 的交互流程是和核心组件息息相关的。 D