【深入理解Java线程池】

深入理解Java线程池

Java线程池是Java并发编程中的一个重要概念,它提供了一种管理和复用线程的机制,可以显著减少创建和销毁线程的开销,提高系统的响应速度和吞吐量。以下是对Java线程池的详细解析:

一、线程池的基本概念

线程池是一个由多个线程组成的集合,这些线程可以被动态地分配给不同的任务。线程池中的线程在执行完一个任务后,并不会被销毁,而是会回到线程池中等待执行下一个任务。这样可以避免频繁地创建和销毁线程,提高系统的性能。

二、线程池的核心参数

  1. corePoolSize(核心线程数):线程池中始终保持的线程数量,即使这些线程处于空闲状态。
  2. maximumPoolSize(最大线程数):线程池能够容纳的最大线程数。当工作队列满时,线程池会尝试创建新的线程,直到线程数量达到maximumPoolSize
  3. keepAliveTime(线程空闲时间):当线程数量超过corePoolSize时,这是多余空闲线程在终止前等待新任务的最长时间。
  4. unit(时间单位):用于指定keepAliveTime的时间单位。
  5. workQueue(任务队列):用于保存等待执行的任务的阻塞队列。
  6. threadFactory(线程工厂):用于创建新线程的工厂。通过线程工厂可以给每个创建出来的线程设置更有意义的名字。
  7. handler(拒绝策略):当队列和线程池都满了,说明线程池处于饱和状态,必须采取一种策略处理提交的新任务。这个策略默认情况下是AbortPolicy,表示无法处理新任务时抛出异常。还可以选择CallerRunsPolicy(用调用者所在线程来运行任务)、DiscardPolicy(丢弃队列里最近的一个任务,并执行当前任务)、DiscardOldestPolicy(丢弃最老的任务,尝试重新提交新的任务)等策略,或者根据应用场景实现RejectedExecutionHandler接口自定义策略。

三、线程池的工作流程

  1. 提交任务:任务通过execute(Runnable command)submit(Callable<T> task)方法提交到线程池。
  2. 判断核心线程是否空闲:如果线程池中的核心线程数量少于corePoolSize,则创建新的核心线程来执行任务。
  3. 放入任务队列:如果核心线程都在忙,则任务会被放入workQueue中等待执行。
  4. 创建非核心线程:如果workQueue已满,且当前线程数量少于maximumPoolSize,则线程池会尝试创建新的非核心线程来执行任务。
  5. 拒绝任务:如果workQueue已满,且线程数量已达到maximumPoolSize,则线程池会根据拒绝策略拒绝新任务。
  6. 线程回收:当线程数量超过corePoolSize,且空闲线程的空闲时间超过keepAliveTime,则这些线程会被回收。

四、常见的线程池类型

  1. FixedThreadPool(固定大小线程池):包含固定数量的线程,适用于需要限制并发线程数量的场景。
  2. CachedThreadPool(缓存线程池):不固定线程数量,可以根据需要自动创建新线程,适用于执行大量短期异步任务。
  3. SingleThreadPool(单线程池):只包含一个工作线程,保证所有任务按顺序执行,适用于需要保持任务顺序执行的场景。
  4. ScheduledThreadPool(定时线程池):可以执行定时任务和周期性任务。
  5. WorkStealingPool(工作窃取线程池):Java 8中引入的一种新类型的线程池,主要用于处理耗时任务,适用于需要大量并行任务、任务之间没有依赖关系的情况。

五、线程池的使用方法和注意事项

  1. 创建线程池:可以使用ThreadPoolExecutor类来创建线程池,并设置相应的参数。也可以使用Executors工厂类来方便地创建常见的线程池类型。
  2. 提交任务:通过execute()submit()方法将任务提交到线程池。
  3. 关闭线程池:线程池应该通过shutdown()shutdownNow()方法显式关闭。shutdown()方法会平滑地关闭线程池,不再接受新任务,但会继续执行队列中的任务。shutdownNow()方法会尝试立即关闭线程池,并尝试停止正在执行的任务。
  4. 注意事项
    • 避免创建过多的线程,合理配置线程池的大小。
    • 任务执行时间不宜过长,以免影响线程池的响应速度。
    • 确保提交给线程池的任务是线程安全的。
    • 根据应用需求合理设置线程池的参数。

六、示例代码

使用ThreadPoolExecutor创建线程池并提交任务的示例代码:

import java.util.concurrent.*;public class ThreadPoolExample {public static void main(String[] args) {// 创建线程池ThreadPoolExecutor executor = new ThreadPoolExecutor(2,  // 核心线程数4,  // 最大线程数60L, TimeUnit.SECONDS,  // 线程空闲时间和时间单位new LinkedBlockingQueue<Runnable>());// 向线程池提交任务for (int i = 0; i < 10; i++) {final int task = i;executor.execute(new Runnable() {@Overridepublic void run() {System.out.println(Thread.currentThread().getName() + " 正在执行第 " + (task + 1) + " 个任务");}});}// 关闭线程池executor.shutdown();}
}

使用Executors工厂类创建不同类型线程池

import java.util.concurrent.*;public class ExecutorsExample {public static void main(String[] args) throws InterruptedException {// 创建一个固定大小的线程池ExecutorService fixedThreadPool = Executors.newFixedThreadPool(3);// 创建一个缓存线程池ExecutorService cachedThreadPool = Executors.newCachedThreadPool();// 创建一个单线程线程池ExecutorService singleThreadPool = Executors.newSingleThreadExecutor();// 创建一个定时线程池ScheduledExecutorService scheduledThreadPool = Executors.newScheduledThreadPool(2);// 创建一个工作窃取线程池ExecutorService workStealingPool = Executors.newWorkStealingPool();// 提交任务给不同类型的线程池for (int i = 0; i < 5; i++) {final int taskIndex = i;fixedThreadPool.submit(() -> printTaskInfo("Fixed", taskIndex));cachedThreadPool.submit(() -> printTaskInfo("Cached", taskIndex));singleThreadPool.submit(() -> printTaskInfo("Single", taskIndex));scheduledThreadPool.schedule(() -> printTaskInfo("Scheduled", taskIndex), 2, TimeUnit.SECONDS); // 延迟2秒执行}// 关闭所有线程池fixedThreadPool.shutdown();cachedThreadPool.shutdown();singleThreadPool.shutdown();scheduledThreadPool.shutdown();workStealingPool.shutdown();}private static void printTaskInfo(String poolType, int taskIndex) {System.out.println(poolType + " Thread Pool: " + Thread.currentThread().getName() + " is executing Task-" + taskIndex);}
}

使用自定义线程工厂和拒绝策略

import java.util.concurrent.*;public class CustomThreadPoolExample {public static void main(String[] args) {// 定义自定义线程工厂ThreadFactory namedThreadFactory = new ThreadFactory() {private int count = 0;@Overridepublic Thread newThread(Runnable r) {return new Thread(r, "MyCustomThread-" + count++);}};// 定义自定义拒绝策略RejectedExecutionHandler customRejectedHandler = new RejectedExecutionHandler() {@Overridepublic void rejectedExecution(Runnable r, ThreadPoolExecutor executor) {System.err.println("Task " + r.toString() + " rejected from " + executor.toString());}};// 创建带有自定义线程工厂和拒绝策略的线程池ThreadPoolExecutor executor = new ThreadPoolExecutor(2, 4, 60L, TimeUnit.SECONDS,new LinkedBlockingQueue<>(10),namedThreadFactory,customRejectedHandler);// 提交大量任务测试自定义拒绝策略for (int i = 0; i < 20; i++) {final int taskNumber = i;executor.execute(() -> printTaskInfo("Custom", taskNumber));}// 关闭线程池executor.shutdown();}private static void printTaskInfo(String poolType, int taskIndex) {System.out.println(poolType + " Thread Pool: " + Thread.currentThread().getName() + " is executing Task-" + taskIndex);}
}

使用submit()方法提交可返回结果的任务

import java.util.concurrent.*;public class CallableExample {public static void main(String[] args) throws InterruptedException, ExecutionException {// 创建线程池ExecutorService executor = Executors.newSingleThreadExecutor();// 提交Callable任务,该任务可以返回结果Future<Integer> future = executor.submit(new Callable<Integer>() {@Overridepublic Integer call() throws Exception {// 模拟耗时操作TimeUnit.SECONDS.sleep(2);return 123;}});// 获取任务的结果try {Integer result = future.get(); // 阻塞直到任务完成System.out.println("Task returned: " + result);} catch (InterruptedException | ExecutionException e) {e.printStackTrace();} finally {// 关闭线程池executor.shutdown();}}
}

使用invokeAll()方法提交多个任务并获取结果

import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.*;public class InvokeAllExample {public static void main(String[] args) throws InterruptedException, ExecutionException {// 创建线程池ExecutorService executor = Executors.newFixedThreadPool(3);// 创建多个Callable任务List<Callable<String>> tasks = new ArrayList<>();for (int i = 0; i < 5; i++) {final int taskId = i;tasks.add(() -> {// 模拟耗时操作TimeUnit.SECONDS.sleep(1);return "Result of Task-" + taskId;});}// 提交所有任务并等待所有任务完成List<Future<String>> futures = executor.invokeAll(tasks);// 获取所有任务的结果for (Future<String> future : futures) {System.out.println(future.get()); // 阻塞直到每个任务完成}// 关闭线程池executor.shutdown();}
}

以上就是对Java线程池的详细解析,包括线程池的基本概念、核心参数、工作流程、常见类型、使用方法和注意事项等内容。通过深入理解线程池,我们可以更好地利用线程池来提高并发应用的性能和可靠性。

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

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

相关文章

KeyFormer:使用注意力分数压缩KV缓存

Keyformer: KV Cache Reduction through Key Tokens Selection for Efficient Generative Inference 202403&#xff0c;发表在Mlsys Introduction 优化KV cache的策略&#xff0c;主要是集中在系统级别的优化上&#xff0c;比如FlashAttention、PagedAttention&#xff0c;它…

Linux 权限管理实践:精确控制用户对 systemctl 和 journalctl 命令的使用

前言 在 Linux 系统管理中&#xff0c;精确控制用户对特定命令的访问权限是一项关键的安全实践。使用 systemctl 和 journalctl 命令时&#xff0c;不当的权限设置可能会导致不必要的风险。本篇博客将详细讨论如何通过 sudoers 文件和 Polkit 策略为不同用户配置 systemctl 和…

SSH连接成功,但VSCode连接不成功

环境 在实验室PC上连接服务器234 解决方案&#xff1a;在VSCode中重新添加远程主机 删除旧的VSCode Server 在远程主机上&#xff0c;VSCode会安装一个‘vscode-server’服务来支持远程开发&#xff0c;有时旧的‘vscode-server’文件可能会导致问题&#xff0c;删除旧的&am…

【Qt】qt安装

在工作一年之后&#xff0c;还是想做一个Qt的教程&#xff0c;遥想研一刚刚接触Qt&#xff0c;从0到1学习&#xff0c;没有什么参考书籍&#xff0c;网上的资料也不多&#xff0c;幸好Qt官方文档写得好&#xff0c;加上自己肯研究&#xff0c;才堪堪入门。 现在我想自己写一个…

Web开发 -前端部分-CSS

CSS CSS&#xff08;Cascading Style Sheet&#xff09;:层叠样式表&#xff0c;用于控制页面的样式&#xff08;表现&#xff09;。 一 基础知识 1 标题格式 标题格式一&#xff1a; 行内样式 <!DOCTYPE html> <html lang"en"><head><meta…

YOLOv8目标检测(六)_封装API接口

YOLOv8目标检测(一)_检测流程梳理&#xff1a;YOLOv8目标检测(一)_检测流程梳理_yolo检测流程-CSDN博客 YOLOv8目标检测(二)_准备数据集&#xff1a;YOLOv8目标检测(二)_准备数据集_yolov8 数据集准备-CSDN博客 YOLOv8目标检测(三)_训练模型&#xff1a;YOLOv8目标检测(三)_训…

MySQL函数—合计统计函数

在MySQL中&#xff0c;你可以使用合计统计函数来计算某个列的合计值、平均值、最大值、最小值等。这些合计统计函数包括SUM、AVG、MAX、MIN等。 下面是一些常用的合计统计函数的示例&#xff1a; SUM函数&#xff1a;用于计算某个列的合计值。 SELECT SUM(column_name) FROM…

51c视觉~YOLO~合集6~

我自己的原文哦~ https://blog.51cto.com/whaosoft/12830685 一、其他yolo 1.1 Spiking-YOLO​ 使用常规深度神经网络到脉冲神经网络转换方法应用于脉冲神经网络域时&#xff0c;性能下降的很多&#xff0c;深入分析后提出了可能的解释&#xff1a;一是来自逐层归一化的效率…

Unity3D 3D模型/动画数据压缩详解

前言 在Unity3D项目中&#xff0c;3D模型和动画数据通常占用大量内存和存储空间&#xff0c;有效的数据压缩技术对于提升游戏性能和加载速度至关重要。本文将详细介绍Unity3D中3D模型和动画数据的压缩技术&#xff0c;并提供相关的代码实现。 对惹&#xff0c;这里有一个游戏…

Elasticsearch Java Api Client中DSL语句的查询方法汇总(二)

接上一篇&#xff1a;《Elasticsearch Java Api Client中DSL语句的查询方法汇总》 说明&#xff1a;示例代码依赖的是co.elastic.clients:elasticsearch-java:8.16.1。 1、ScriptQuery方法 用途&#xff1a;它允许用户使用脚本&#xff08;通常是 Painless 脚本语言&#xf…

如何在 Ubuntu 22.04 上安装 Strapi CMS

简介 Strapi 是一个使用 JavaScript 构建的开源、无头内容管理系统 (CMS)。与其他无头 CMS 一样&#xff0c;Strapi 开箱即用不带前端。它使用 API 作为其前端&#xff0c;允许你使用流行的框架&#xff08;如 React 和 Next.js&#xff09;构建网站。Strapi 基于插件系统&…

数字IC后端零基础入门基础理论(Day1)

数字IC后端设计导入需要用到的input数据如下图所示。 数字后端零基础入门系列 | Innovus零基础LAB学习Day9 Netlist: 设计的Gate level&#xff08;门级&#xff09;网表。下图所示为一个计数器设计综合后的门级netlist。 从这个netlist中我们看到这个设计顶层的名字叫counte…

序列模型的使用示例

序列模型的使用示例 1 RNN原理1.1 序列模型的输入输出1.2 循环神经网络&#xff08;RNN&#xff09;1.3 RNN的公式表示2 数据的尺寸 3 PyTorch中查看RNN的参数4 PyTorch中实现RNN&#xff08;1&#xff09;RNN实例化&#xff08;2&#xff09;forward函数&#xff08;3&#xf…

如何评估呼叫中心大模型呼出机器人的使用效果?

如何评估呼叫中心大模型呼出机器人的使用效果&#xff1f; 原作者&#xff1a;开源呼叫中心FreeIPCC&#xff0c;其Github&#xff1a;https://github.com/lihaiya/freeipcc 评估呼叫中心大模型呼出机器人的使用效果是一个复杂而多维的过程&#xff0c;需要综合考虑多个方面&…

WSL2内部的Ubuntu怎么设置网络内桥接模式,弄了好久老是不成功,怎么办?

环境: Win10专业版 WSL2 Ubuntu22.04 问题描述: WSL2内部的Ubuntu怎么设置网络内桥接模式 解决方案: 方法一 1.控制面板开启,Hyper-V 管理器 2.重启电脑 3…创建外部虚拟交换机 打开 Hyper-V 管理器,在右侧操作面板中点击“虚拟交换机管理器”。 选择“创建虚…

redis集群 服务器更换ip,怎么办,怎么更换redis集群的ip

redis集群 服务器更换ip&#xff0c;怎么办&#xff0c;怎么更换redis集群的ip 1、安装redis三主三从集群2、正常状态的redis集群3、更改redis集群服务器的ip 重启服务器 集群会down4、更改redis集群服务器的ip 重启服务器 集群down的原因5、更改redis集群服务器的ip后&#xf…

记录学习《手动学习深度学习》这本书的笔记(五)

这一章是循环神经网络&#xff0c;太难了太难了&#xff0c;有很多卡壳的地方理解了好久&#xff0c;比如隐藏层和隐状态的区别、代码的含义&#xff08;为此专门另写了一篇【笔记】记录对自主实现一个神经网络的步骤的理解&#xff09;、梯度计算相关&#xff08;【笔记】记录…

人大金仓数据linux安装注意事项

人大金仓数据linux安装注意事项 本次是个人搭建虚拟机安装centos7的环境下进行安装。 1、安装流程参照https://help.kingbase.com.cn/v9/install-updata/install-linux/preface.html。 2、mount安装文件报错 操作手册提供mount的命令如下&#xff1a; mount KingbaseES_V009R0…

【GIS教程】使用GDAL-Python将tif转为COG并在ArcGIS Js前端加载-附完整代码

目录 一、数据格式 二、COG特点 三、使用GDAL生成COG格式的数据 四、使用ArcGIS Maps SDK for JavaScript加载COG格式数据 一、数据格式 COG&#xff08;Cloud optimized GeoTIFF&#xff09;是一种GeoTiff格式的数据。托管在 HTTP 文件服务器上&#xff0c;可以代替geose…

探索智能时代:如何利用AI一键生成PPT改变演示文稿的制作方式

在这个科技飞速发展的时代&#xff0c;信息的传递方式发生了翻天覆地的变化。曾几何时&#xff0c;我们还在为制作PPT而熬夜&#xff0c;手动选择模板、调整布局&#xff0c;甚至为每一张幻灯片的内容苦思冥想。然而&#xff0c;随着人工智能技术的不断进步&#xff0c;制作PPT…