Java 线程池详解

序言

在高并发编程中,线程池是一个非常重要的组件。它不仅能够有效地管理和复用线程资源,还可以提升应用程序的性能和稳定性。本文将详细介绍Java中的线程池机制,以及如何正确地使用线程池。

一、什么是线程池

线程池是一组已经初始化并等待执行任务的线程集合。通过使用线程池,我们可以避免频繁地创建和销毁线程,从而节省资源和减少系统开销。线程池的核心思想是通过复用线程来提高性能。

二、为什么使用线程池

  • 资源消耗:通过池化技术重复利用已创建的线程,降低线程创建和销毁造成的损耗。
  • 提高响应速度:任务到达时,无需等待线程创建即可立即执行。
  • 提高线程的可管理性:线程是稀缺资源,如果无限制创建,不仅会消耗系统资源,还会因为线程的不合理分布导致资源调度失衡,降低系统的稳定性。使用线程池可以进行统一的分配、调优和监控。

三、Java 中的线程池

Java 提供了 java.util.concurrent 包来支持并发编程。其中有几种方法创建线程池

1、通过 Executors 工厂类的静态方法创建线程池:

newFixedThreadPool
 /*** 创建一个可重用固定数量线程的线程池,这些线程操作于共享的无界队列。* 在任何时候,最多有 {@code nThreads} 个线程会处于活动状态处理任务。* 当所有线程都在活动时,如果提交了额外的任务,它们将在队列中等待,* 直到有线程可用。* 如果在关闭前,任何线程因执行失败而终止,将根据需要创建一个新的线程来执行后续任务。* 线程池中的线程将持续存在,直到显式调用 {@link ExecutorService#shutdown shutdown} 方法关闭它。** @param nThreads 线程池中的线程数量* @return 新创建的线程池* @throws IllegalArgumentException 如果 {@code nThreads <= 0}*/public static ExecutorService newFixedThreadPool(int nThreads) {return new ThreadPoolExecutor(nThreads, nThreads,0L, TimeUnit.MILLISECONDS,new LinkedBlockingQueue<Runnable>());}
newSingleThreadExecutor

/*** 创建一个使用单一工作线程的ExecutorService来执行任务。* 此执行器服务使用单一线程以顺序方式处理任务,确保每次只执行一个任务。* 即使在执行前一个任务时因失败导致线程终止,在关闭前也会创建新的线程继续执行后续任务。* 与等效的{@code newFixedThreadPool(1)}不同,返回的执行器保证不能重新配置为使用额外的线程。** @return 新创建的单线程ExecutorService*/
public static ExecutorService newSingleThreadExecutor() {return new FinalizableDelegatedExecutorService(new ThreadPoolExecutor(1, 1,0L, TimeUnit.MILLISECONDS,new LinkedBlockingQueue<Runnable>()));
}
其中这两个线程池里面使用到的阻塞队列如下:
// 队列的最大长度是Integer的最大整数
public LinkedBlockingQueue() {this(Integer.MAX_VALUE);
}
newCachedThreadPool
/*** 创建一个可根据需要创建新线程的线程池,同时会重用之前已构建的空闲线程。* 这种线程池通常能提升执行大量短生命周期异步任务程序的性能。* 调用{@code execute}方法时,如果存在可用的先前构建的线程则会重用它们。* 若无现有线程可用,将创建新线程并加入到线程池中。* 在线程闲置六十秒后,线程将被终止并从缓存中移除。* 因此,长时间处于空闲状态的线程池不会消耗任何资源。* 注意:可以通过{@link ThreadPoolExecutor}构造器创建具有类似特性但细节不同的(例如超时参数)线程池。** @return 新创建的线程池实例*/
public static ExecutorService newCachedThreadPool() {return new ThreadPoolExecutor(0, Integer.MAX_VALUE,60L, TimeUnit.SECONDS,new SynchronousQueue<Runnable>());
}

注:一般在工程上不建议使用它们创建线程池,阿里java开发手册

在这里插入图片描述

2、通过 ScheduledExecutorService 创建定时任务线程池:

ScheduledExecutorService 是 ExecutorService 的子接口,专门用于支持定时及周期性任务执行的线程池。可以通过 Executors.newScheduledThreadPool(int corePoolSize) 方法创建:

 /*** 创建一个可以调度命令在指定延迟后运行,或周期性执行的线程池。* * @param corePoolSize 即使在空闲时,线程池中也要保持的线程数量* @return 新创建的ScheduledExecutorService实例,即计划线程池* @throws IllegalArgumentException 如果corePoolSize小于0*/
public static ScheduledExecutorService newScheduledThreadPool(int corePoolSize) {return new ScheduledThreadPoolExecutor(corePoolSize);
}public ScheduledThreadPoolExecutor(int corePoolSize) {super(corePoolSize, Integer.MAX_VALUE,DEFAULT_KEEPALIVE_MILLIS, MILLISECONDS,new DelayedWorkQueue());
}

3、通过 ThreadPoolExecutor 构造函数直接创建:

使用 ThreadPoolExecutor 的构造函数可以更加灵活地配置线程池,例如指定核心线程数、最大线程数、线程存活时间、工作队列等参数,如下所示:

 public ThreadPoolExecutor(int corePoolSize, // 核心线程数int maximumPoolSize, // 最大线程数long keepAliveTime, // 线程存活时间TimeUnit unit, // 线程存活时间单位BlockingQueue<Runnable> workQueue, // 阻塞队列ThreadFactory threadFactory, // 线程工厂RejectedExecutionHandler handler //拒绝策略
)

四、ThreadPoolExecutor 线程池的执行过程

流程图是这样的:
在这里插入图片描述
整体流程就是:

1、任务来的时候,如果当前线程数小于核心线程数,则创建线程执行任务
2、否则,将任务添加到阻塞队列
3、如果队列已满case1:如果当前线程数小于最大线程数,则创建线程执行任务case2:如果当前线程数大于等于最大线程数,则执行拒绝策略

其实这个流程整体上来说是没什么问题的,也是大多数面试者回答该问题的标准答案。
但是这里有一个小问题,就是当我们把核心线程数设置为0的时候,声明的阻塞队列长度随便给一个大于0的值即可(此处我使用LinkedBlockingQueue队列,它的默认长度是Integer的最大值)。
我们来看看下面的代码,大家觉得会输出什么?

public static void testThreadPool() {ThreadPoolExecutor threadPoolExecutor = new ThreadPoolExecutor(0, 10, 60, TimeUnit.SECONDS, new LinkedBlockingQueue<>(),Executors.defaultThreadFactory(), new ThreadPoolExecutor.AbortPolicy());threadPoolExecutor.execute(()->{System.out.println("线程池测试: " + new Date());});
}

如果按照上面流程图的步骤进行的话,什么也不会输出,因为此任务会加入到阻塞队列里面,对不对?
其实它输出了,结果如下
在这里插入图片描述
你现在可能有一个大大问号,其实我们可以从源码那里得到答案,源码如下:

public void execute(Runnable command) {// 检查传入任务是否为空,为空抛出异常if (command == null)throw new NullPointerException();// 获取当前线程池控制状态int c = ctl.get();// 1、如果当前线程数小于核心线程数if (workerCountOf(c) < corePoolSize) {// 添加一个新的工作线程来执行任务// 添加成功,直接返回if (addWorker(command, true))return;// 再次获取线程池控制状态,防止并发c = ctl.get();}// 2、如果线程池处于运行状态,并且任务能够加入队列if (isRunning(c) && workQueue.offer(command)) {// 再次获取线程池控制状态,防止并发int recheck = ctl.get();// 2.1、如果线程池不再运行,并且任务能够从队列中移除,则拒绝任务if (! isRunning(recheck) && remove(command))reject(command);// 2.2、如果当前线程池没有运行的线程(此处可以打消你的问号)else if (workerCountOf(recheck) == 0)// 添加一个新的非核心工作线程addWorker(null, false);}// 3、添加一个新的非核心工作线程,如果失败,拒绝任务else if (!addWorker(command, false))reject(command);
}

所以说,从源码的角度具体去分析线程池执行任务的流程如下:

1、任务来的时候,如果当前线程数小于核心线程数,则创建线程执行任务
2、否则,线程池处于运行状态并且任务能够加入队列case1:判断线程池运行状态,如果线程池状态不是运行的情况下,同时任务可以从队列移除,那么就拒绝该任务case2:判断当前是否有工作线程,如果没有,则创建一个非核心的工作线程
3、如果队列已满case1:如果当前线程数小于最大线程数,则创建线程执行任务case2:如果当前线程数大于等于最大线程数,则执行拒绝策略

五、线程池的拒绝策略

名称描述
AbortPolicy丢弃任务并抛出RejectedExecutionException异常。这是线程池默认的拒绝策略,在任务不能再提交的时候,抛出异常,及时反馈程序运行状态。如果是比较关键的业务推荐使用此拒绝策略,这样子在系统不能承载更大的并发量的时候,能够及时的通过异常发现。
CallerRunsPolicy由调用线程(提交任务的线程)处理该任务。这种情况是需要让所有任务都执行完毕那么就适合大量计算的任务类型去执行,多线程仅仅是增大吞吐量的手段,最终必须要让每个任务都执行完毕。
DiscardPolicy丢弃任务,但是不抛出异常。 使用此策略,可能会使我们无法发现系统的异常状态。建议是一些无关紧要的业务采用此策略。
DiscardOldestPolicy丢弃队列最前面的任务,然后重新提交被拒绝的任务。是否要采用此种拒绝策略,还得根据实际业务是否允许丢弃老任务来认真衡量。

六、线程池提交任务

线程池有两种提交任务的方式,

1、使用submit(Runnable task),有返回值

public static void testThreadPool() {ThreadPoolExecutor threadPoolExecutor = new ThreadPoolExecutor(0, 10, 60,TimeUnit.SECONDS,new LinkedBlockingQueue<>(),Executors.defaultThreadFactory(),new ThreadPoolExecutor.AbortPolicy());Future<?> submit = threadPoolExecutor.submit(() -> {System.out.println("线程池测试submit: " + new Date());return 1;});try {Object o = submit.get();System.out.println(o);} catch (InterruptedException e) {throw new RuntimeException(e);} catch (ExecutionException e) {throw new RuntimeException(e);}
}

2、使用execute(Runnable task),没有返回值

public static void testThreadPool() {ThreadPoolExecutor threadPoolExecutor = new ThreadPoolExecutor(0, 10, 60,TimeUnit.SECONDS,new LinkedBlockingQueue<>(),Executors.defaultThreadFactory(),new ThreadPoolExecutor.AbortPolicy());threadPoolExecutor.execute(()->{System.out.println("线程池测试execute: " + new Date());});
}

七、线程池的状态

名称描述
RUNNING会接收新任务并且会处理队列中的任务
SHUTDOWN不会接收新任务并且会处理队列中的任务,任务处理完后中断所有线程
STOP不会接收新任务并且不会处理队列中的任务,直接中断所有线程
TIDYING所有任务都已终止,workerCount为0
TERMINATEDterminated()执行完成之后就会转变为TERMINATED

这五种状态之间的转换

RUNNING -> SHUTDOWN:调用shutdown方法RUNNING or SHUTDOWN -> STOP:调用shutdownNow方法SHUTDOWN -> TIDYING:阻塞队列为空,线程池中工作线程数为0STOP-> TIDYING:线程池中工作线程数为0TIDYING -> TERMINATED:执行erminated方法

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/871586.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

索引结构与检索原理

一、mysql索引结构 1.BTree索引 [检索原理] 左边列的表格&#xff08;真实数据&#xff09;&#xff0c;右边对应一棵树&#xff0c;树的管度越来越管查询越快。 以下图表的名称为&#xff1a;段区块 硬盘都是长方形的&#xff0c;打了一个封装&#xff0c;里面是一个圆圈…

二分查找和斐波那契查找

这里写自定义目录标题 二分查找斐波那契查找二分查找改进B二分查找改进C 二分查找 int binSearch(int* arr, int lo, int hi,int target) {while (lo < hi){int mid lo ((hi - lo) >> 1);if (arr[mid] > target) hi mid;else if (arr[mid] < target) lo mi…

springBoot 核心原理

自动配置 包扫描规则&#xff1a; 默认的扫描规则 SpringBootApplication 标注的类就是主程序类 &#xff0c;springBoot也只会扫描主程序类所在的包以及下面的子包也可以自定义声明扫描其他包 使用 SpringBootApplication(scanBasePackages “com.test”)使用 ComponentSca…

excel、word、ppt 下载安装步骤整理

请按照我的步骤开始操作&#xff0c;注意以下截图红框标记处&#xff08;往往都是需要点击的地方&#xff09; 第一步&#xff1a;下载 首先进入office下载网址&#xff1a; otp.landian.vip 然后点击下载 拉到下方 下载站点&#xff08;这里根据自己的需要选择下载&#x…

汇编语言程序设计-7-高级汇编语言技术

7. 高级汇编语言技术 文章目录 7. 高级汇编语言技术7.0 导学7.1 子程序的另外一种写法-segment/ends-proc/endp7.2 程序的多文件组织7.3 汇编指令汇总7.4 汇编伪操作汇总7.5 汇编操作符汇总7.6 汇编过程7.7 宏汇编7.8 宏库7.9 条件汇编7.10 重复汇编7.11 80x86汇编7.12 汇编语言…

NAT地址转换+多出口智能选路,附加实验内容

本章主要讲&#xff1a;基于目标IP、双向地址的转换 注意&#xff1a;基于目标NAT进行转换 ---基于目标IP进行地址转换一般是应用在服务器端口映射&#xff1b; NAT的基础知识 1、服务器映射 服务器映射是基于目标端口进行转换&#xff0c;同时端口号也可以进行修改&…

方波的傅里叶变换及方波的MATLAB实现

一、傅里叶变换简介 傅里叶变换&#xff0c;表示能将满足一定条件的某个函数表示成三角函数&#xff08;正弦和/或余弦函数&#xff09;或者它们的积分的线性组合。傅里叶变换是一种线性的积分变换。它的理论依据是&#xff1a;任何连续周期信号都可以由一组适当的正弦曲线组合…

stm32h743 NetXduo 实现http server CubeIDE+CubeMX

在这边要设置mpu的大小,要用到http server,mpu得设置的大一些 我是这么设置的,做一个参考 同样,在FLASH.ld里面也要对应修改,SECTIONS里增加.tcp_sec和 .nx_data两个区,我们用ram_d2区域去做网络,这个就是对应每个数据在d2区域的起点。 在CubeMX里,需要用到filex、dhc…

无损音乐播放器推荐:Audirvana for Mac 中文激活版

udirvana 是一款高品质的音乐播放软件&#xff0c;专为Mac操作系统设计。它被设计来提供音频播放的最高标准&#xff0c;支持多种音频格式&#xff0c;包括高达32位/192kHz的高分辨率音频。Audirvana Plus 是其高级版本&#xff0c;提供了更多的功能和优化&#xff0c;例如音频…

Qt Mqtt客户端 + Emqx

环境 Qt 5.14.2 qtmqtt mqttx 功能 QT Mqtt客户端 qtmqtt 下载 qtmqtt (注意下载与QT版本相符的库)并使用QT 编译 编译完成后需要的文件: emqx 1.虚拟机中安装emqx,并启动 curl -s https://assets.emqx.com/scripts/install-emqx-deb.sh | sudo bash sudo apt-get inst…

前端Vue组件化实践:打造仿京东天猫商品属性选择器组件

在前端开发领域&#xff0c;随着业务需求的日益复杂和技术的不断进步&#xff0c;传统的整体式应用开发模式已逐渐显得捉襟见肘。面对日益庞大的系统&#xff0c;每次微小的功能修改或增加都可能导致整个逻辑结构的重构&#xff0c;形成牵一发而动全身的困境。为了解决这一问题…

树莓派PICO使用INA226测量电流和总线电压(3)

上一篇文章我们讲了如何测试电流&#xff0c;但是INA226有一个非常典型的问题&#xff0c;那就是误差比较大&#xff0c;因为采样电阻非常小&#xff0c;我的开发板用的是100mΩ的采样电阻&#xff0c;在设定中我也用的是这个采样电阻值&#xff0c;但事实上&#xff0c;测试得…

免费开源工具—— Clarity Al:一键图像放大/增强,Magnific平替!

今天给大家推荐一款图像增强工具——Clarity AI &#xff0c;免费且开源&#xff0c;快来看看吧&#xff01; 1、效果展示 MagnificAl是一款基于人工智能技术的图像处理工具,主要功能包括图像放大、像素级AI重绘、灵活的设置调整以及多种优化场景。它能够支持最高放大至16倍,甚…

算法力扣刷题记录 四十五【110.平衡二叉树】

前言 二叉树篇继续 记录 四十五【110.平衡二叉树】 一、题目阅读 给定一个二叉树&#xff0c;判断它是否是 平衡二叉树。 示例 1&#xff1a; 输入&#xff1a;root [3,9,20,null,null,15,7] 输出&#xff1a;true示例 2&#xff1a; 输入&#xff1a;root [1,2,2,3,3…

电容充放电时间计算

电容充电时间的结论&#xff1a;t充电 R * C 时&#xff0c;Ut2*VCC/3&#xff0c;这是一个不能让我释怀的结论。 1、电池充电 U0表示电容C在充电0时刻的电压值; Ut表示电容C在充电t时刻的电压值; U1表示电容C在充电∝时刻的电压值&#xff0c;就是电源电压; Q C * U ---…

【C/C++积累技巧】实现 连续播放文件图片+逐帧文本显示, 同时 可以按任意键退出(基于easyx小游戏编程)

技巧一、使用 IMAGE数组循环&#xff1a;实现【连续播放图片】 &#xff08;1&#xff09;一张图片如何放映在 图形化窗口上&#xff1a;借用两个函数 #include<graphics.h> // 函数的头文件IMAGE imgMy; // 图形变量 loadimage(&imgMy, "写入你想显示的图片路…

Java高频面试基础知识点整理27

干货分享&#xff0c;感谢您的阅读&#xff01;背景​​​​​​高频面试题基本总结回顾&#xff08;含笔试高频算法整理&#xff09; 最全文章见&#xff1a;Java高频面试基础知识点整理 &#xff08;一&#xff09;Java基础高频知识考点 针对人员&#xff1a; 1.全部人员都…

vs2019 QT无法打开源文件QModbusTcpClient

vs2019无法打开源文件QModbusTcpClient 如果配置的msvc2019,则查找到Include目录 然后包含&#xff1a; #include <QtSerialBus/qmodbustcpclient.h>

STL 提供的容器可以有多快?(下)「榨干最后一滴」

以下内容为本人的烂笔头&#xff0c;如需要转载&#xff0c;请声明原文链接 微信公众号「ENG八戒」https://mp.weixin.qq.com/s/QWgA97TDMGBnwR4hKA7BwA 查表的消耗 某些场景下需要用到大量的 (string, X) 键值对来存储数据&#xff0c;标准库提供了关联容器 std::map 来解决键…

Python酷库之旅-第三方库Pandas(021)

目录 一、用法精讲 52、pandas.from_dummies函数 52-1、语法 52-2、参数 52-3、功能 52-4、返回值 52-5、说明 52-6、用法 52-6-1、数据准备 52-6-2、代码示例 52-6-3、结果输出 53、pandas.factorize函数 53-1、语法 53-2、参数 53-3、功能 53-4、返回值 53-…