Java 线程池之 ThreadPoolExecutor

        Java线程池,特别是ThreadPoolExecutor,是构建高性能、可扩展应用程序的基石之一。它不仅关乎效率,还直接关系到资源管理与系统稳定性。想象一下,如果每来一个请求就创建一个新的线程,服务器怕是很快就要举白旗了。而ThreadPoolExecutor就是那个懂得“量入为出”,合理调配资源的智慧管家。

详细介绍

        ThreadPoolExecutor是Java并发编程中线程池的核心实现类,它位于java.util.concurrent包下,由Doug Lea设计,是Java 1.5引入的重要特性之一。ThreadPoolExecutor提供了一种灵活的方式来管理和控制线程的创建、执行、调度和回收,以高效地执行大量异步任务。

  ThreadPoolExecutor实现了ExecutorService接口,提供了强大的线程池管理功能。它的构造方法允许高度定制,主要包括核心线程数、最大线程数、线程空闲时间、任务队列、拒绝策略等关键参数。

ThreadPoolExecutor(int corePoolSize, 
                  int maximumPoolSize, 
                  long keepAliveTime, 
                  TimeUnit unit, 
                  BlockingQueue<Runnable> workQueue, 
                  ThreadFactory threadFactory, 
                  RejectedExecutionHandler handler)
  • corePoolSize:线程池的基本大小,即使没有任务执行,也会保持这么多线程。
  • maximumPoolSize:线程池最大线程数,当队列满且工作线程数小于最大值时,会创建新线程执行任务。
  • keepAliveTime:非核心线程闲置时的超时时长,超过这个时间会被回收。
  • unitkeepAliveTime的时间单位。
  • workQueue:用于保存待处理任务的阻塞队列,有多种队列类型可以选择,如ArrayBlockingQueueLinkedBlockingQueueSynchronousQueue等。
  • threadFactory:线程工厂,用来创建新线程。
  • handler:拒绝策略处理器,当线程池和队列都满时,如何处理新提交的任务。

使用场景

  • 高并发请求处理:如Web服务器处理大量HTTP请求。
  • 批量数据处理:如图像处理、文件操作等大量IO操作。
  • 定时任务执行:结合ScheduledThreadPoolExecutor实现定时任务。
  • 并行计算:科学计算、大数据处理等需要并行处理的场景。

Java代码示例

import java.util.concurrent.*;

public class ThreadPoolDemo {
    public static void main(String[] args) {
        // 创建线程池
        ThreadPoolExecutor executor = new ThreadPoolExecutor(
                5, // 核心线程数
                10, // 最大线程数
                1, // 空闲线程存活时间,单位秒
                TimeUnit.SECONDS,
                new LinkedBlockingQueue<>(20), // 任务队列
                Executors.defaultThreadFactory(), // 线程工厂
                new ThreadPoolExecutor.AbortPolicy() // 拒绝策略,超出时抛出异常
        );
        
        // 提交任务
        for (int i = 0; i < 30; i++) {
            int taskId = i;
            executor.execute(() -> downloadTask(taskId));
        }
        
        // 关闭线程池
        executor.shutdown();
        while (!executor.isTerminated()) {
            // 等待所有任务完成
        }
        System.out.println("所有下载任务已完成!");
    }
    
    private static void downloadTask(int taskId) {
        System.out.println("开始下载任务 " + taskId);
        try {
            Thread.sleep(1000); // 模拟下载耗时
        } catch (InterruptedException e) {
            Thread.currentThread().interrupt();
        }
        System.out.println("任务 " + taskId + " 下载完成!");
    }
}

实际开发中的使用和注意事项

  1. 合理配置线程池参数:根据任务类型和系统资源合理设置线程池大小、队列长度等。
  2. 避免任务无限提交:确保任务提交策略不会导致内存溢出。
  3. 资源回收:使用完毕后调用shutdown()shutdownNow(),确保线程池和资源正确释放。
  4. 监控线程池状态:定期检查线程池状态,如使用getPoolSize()getActiveCount()等方法。
  5. 异常处理:实现或自定义RejectedExecutionHandler来妥善处理拒绝的任务。

优缺点

优点

  • 提高性能:减少线程创建和销毁的开销。
  • 资源管理:有效控制线程数量,避免资源耗尽。
  • 灵活性:通过自定义参数满足不同场景需求。

缺点

  • 配置复杂:需要对业务和系统有深入理解才能合理配置。
  • 调试难度:线程池内部错误和异常处理相对复杂。
  • 滥用风险:不恰当使用可能导致性能下降或资源耗尽。

可能遇到的问题及解决方案

1. 任务积压

问题描述:当线程池中线程已达到最大数,且任务队列也已满载时,新提交的任务将面临被拒绝的风险,导致任务积压在应用层面,可能引起内存溢出或响应延迟。

解决方案

  • 调整队列大小:适当增加队列容量,但这会增加内存占用,且只是临时解决方案。
  • 动态调整最大线程数:根据系统负载动态调整maximumPoolSize,使用如ThreadPoolExecutor.CallerRunsPolicy拒绝策略让调用者线程直接执行任务,减轻队列压力。
  • 使用有界队列:如ArrayBlockingQueue,确保队列不会无限增长,迫使线程池拒绝超出队列容量的任务。
  • 监控与报警:建立线程池和任务队列的监控机制,当接近阈值时触发报警,以便及时调整策略或扩容。

2. 线程泄漏

问题描述:如果任务执行过程中抛出了未捕获的异常,或者finally块中的资源清理代码未能正确执行,可能导致线程无法正确回收,进而造成线程池“泄漏”。

解决方案

  • 确保异常处理:在RunnableCallable的任务实现中,使用try-catch-finally结构,确保异常被捕获且资源得到正确释放。
  • 自定义异常处理器:实现或定制Thread.UncaughtExceptionHandler,捕获并处理未捕获的异常,必要时记录日志并尝试恢复。
  • 使用守护线程:确保ThreadPoolExecutor的线程是守护线程(通过ThreadFactory设置),这样当主线程结束时,即使有线程未正确终止,JVM也会退出。

3. 拒绝策略处理不当

问题描述:当线程池和任务队列都达到饱和状态时,预设的拒绝策略(如默认的AbortPolicy)会直接抛出RejectedExecutionException,如果不做特殊处理,可能导致任务丢失。

解决方案

  • 选择合适的拒绝策略:根据业务需求选择或自定义拒绝策略,如CallerRunsPolicy让调用者线程执行任务,DiscardPolicy丢弃任务不抛出异常,或DiscardOldestPolicy丢弃最旧的任务来接纳新任务。
  • 二次提交或重试机制:捕获RejectedExecutionException后,可以根据情况设计任务的重试逻辑,或者将任务放入备用队列稍后重试。
  • 动态扩容:在检测到拒绝策略触发时,考虑是否可以动态增加线程池的容量,但需谨慎,避免无限制增长。

4. 死锁

问题描述:在多线程环境下,如果任务间存在不当的同步或等待关系,可能导致死锁,即两个或多个线程互相等待对方释放资源,无法继续执行。

解决方案

  • 避免嵌套锁:尽量减少锁的使用,避免在持有锁的同时去获取另一个锁。
  • 按照相同的顺序获取锁:如果必须使用多个锁,确保所有线程按照相同的顺序获取锁。
  • 超时机制:在尝试获取锁时使用带有超时的尝试锁定方法,如tryLock(long time, TimeUnit unit),避免无限等待。
  • 检测与诊断:使用Java的jstack命令或集成监控工具定期检查死锁情况,一旦发现,通过日志分析或人工介入解决。

        通过深入理解和恰当应用ThreadPoolExecutor,我们不仅能提升应用的性能,还能确保系统的稳定性和可维护性。合理使用和维护ThreadPoolExecutor是保证Java应用并发性能的关键。面对问题,采取恰当的预防和应对措施至关重要,当然持续的监控与调优也是必不可少的环节。通过这些方法,可以有效提升应用的稳定性和效率。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Web Component fancy-components

css-doodle 组件库 fancy-components 组件库使用 yarn add fancy-components使用&#xff1a; import { FcBubbles } from fancy-components new FcBubbles() //要用哪个就new哪个 new 这里可能会报错eslink,eslintrc.js中处理报错 module.exports {rules: {no-new: off} …

【智能算法应用】基于麻雀搜索算法的二维最大熵图像阈值分割

目录 1.算法原理2.数学模型3.结果展示4.参考文献5.代码获取 1.算法原理 【智能算法】麻雀搜索算法&#xff08;SSA&#xff09;原理及实现 2.数学模型 最大熵法是由 Kapur 于 1985 年所提出的&#xff0c; 该方法的阈值选取标准取决于图像中最大化分 割的目标区域和背景区域…

初学java

注意点 1.使用关键字long的时候&#xff0c;在其赋值的时候要在后面加上大写或者小写的l&#xff0c;个人推荐大写&#xff0c;小写与数‘1’难区分。 2.函数的名字要与文件夹的名字相同&#xff0c;并且文件夹后面一定要有.java。例如这个的名字是Main,函数就得用这个&#x…

python+pycharm安装教程

介绍 Python提供了高效的高级数据结构&#xff0c;还能简单有效地面向对象编程。Python语法和动态类型&#xff0c;以及解释型语言的本质&#xff0c;使它成为多数平台上写脚本和快速开发应用的编程语言&#xff0c;Python解释器易于扩展&#xff0c;可以使用C、C或其他可以通过…

国科大深度学习期末历年试卷

本文借鉴 国科大深度学习复习 深度学习期末 深度学习2020 一&#xff0e;名词解释&#xff08;每个2分&#xff0c;共10分&#xff09; 深度学习&#xff0c;稀疏自编码器&#xff0c;正则化&#xff0c;集成学习&#xff0c;Dropout 二&#xff0e;简答题&#xff08;每题…

Autoxjs 实践-Spring Boot 集成 WebSocket

概述 最近弄了福袋工具&#xff0c;由于工具运行中&#xff0c;不好查看福袋结果&#xff0c;所以我想将福袋工具运行数据返回到后台&#xff0c;做数据统计、之后工具会越来越多&#xff0c;就弄了个后台&#xff0c;方便管理。 实现效果 WebSocket&#xff1f; websocket是…

机器学习:基于TF-IDF算法、决策树,使用NLTK库对亚马逊美食评论进行情绪分析

前言 系列专栏&#xff1a;机器学习&#xff1a;高级应用与实践【项目实战100】【2024】✨︎ 在本专栏中不仅包含一些适合初学者的最新机器学习项目&#xff0c;每个项目都处理一组不同的问题&#xff0c;包括监督和无监督学习、分类、回归和聚类&#xff0c;而且涉及创建深度学…

论文阅读】 ICCV-2021-3D Local Convolutional Neural Networks for Gait Recognition

motivation :现有方法方法无法准确定位身体部位&#xff0c;不同的身体部位可以出现在同一个条纹(如手臂和躯干)&#xff0c;一个部分可以出现在不同帧(如手)的不同条纹上。其次&#xff0c;不同的身体部位具有不同的尺度&#xff0c;即使是不同帧中的同一部分也可以出现在不同…

【光速上手 Hydra 】一行代码自动跑多次实验,Hydra 中的 Multirun 参数如何使用?

Hydra 是一个开源的 Python 框架&#xff0c;简化了研究和其他复杂应用的开发。其关键特性是能够通过组合动态地创建一个分层次的配置&#xff0c;并通过配置文件和命令行进行覆盖。Hydra 的名称来源于其能够运行多个类似的作业 - 就像一个有多个头的九头蛇一样。 主要特性&am…

泽众财务RPA机器人常见五个应用场景

泽众RPA&#xff08;即机器人流程自动化&#xff0c;Robotic Process Automation, RPA&#xff09;解决方案是依托于各类先进信息技术手段的虚拟劳动力 &#xff08;数字劳动力&#xff09;&#xff0c;根据预先设定的程序操作指令对任务进行自动化处理&#xff0c;实现业务流程…

解锁机械之美:发动机设备拆解可视化揭秘

在现代工程技术的世界里&#xff0c;发动机作为机械设备的心脏&#xff0c;始终扮演着至关重要的角色。图扑的发动机设备拆解可视化技术&#xff0c;以其独特的视角和精确的细节&#xff0c;为我们开启了一扇了解复杂机械结构的新窗。通过高清晰度的三维图像和动画&#xff0c;…

在阿里云K8S容器中,部署websocket应用程序的总结

一、背景 有一个websocket应用程序&#xff0c;使用spring boot框架开发&#xff0c;http端口号是6005&#xff0c;提供的是websocket服务&#xff0c;所以它还监听一个8889端口的tcp协议。 现在要把它部署到阿里云的k8s容器里&#xff0c;本文着重描述service层的配置。 因…

鸿蒙开发接口Ability框架:【 (ServiceExtensionAbility)】

ServiceExtensionAbility ServiceExtensionAbility模块提供ServiceExtension服务扩展相关接口的能力。 说明&#xff1a; 本模块首批接口从API version 9开始支持。后续版本的新增接口&#xff0c;采用上角标单独标记接口的起始版本。 本模块接口仅可在Stage模型下使用。 导入…

【linux软件基础知识】-死锁问题

死锁问题 当两个或多个线程由于每个线程都在等待另一个线程持有的资源而无法继续时,就会发生死锁 如下图所示, 在线程 1 中,代码持有了 L1 上的锁,然后尝试获取 L2 上的锁。 在线程 2 中,代码持有了 L2 上的锁,然后尝试获取 L1 上的锁。 在这种情况下,线程 1 已获取 L…

减速机齿数速算

1.齿轮相关参数 1.1 模数 &#xff0c; 因为 齿数*齿距 Pi*直径 所以&#xff1a;直径/齿数 齿距/PI 模数 国标现行标准&#xff08;截止2024/5&#xff09;是&#xff1a; GB/ 1357-2008 / ISO 54-1996 模数有国标的一个序列标准&#xff1a; 1.2.轴径 轴径的国标是&a…

【测试报告】星光日册

⭐ 作者&#xff1a;Jwenen &#x1f331; 作者主页&#xff1a;Jwenen的个人主页 &#x1f496; 持续更文&#xff0c;关注博主少走弯路&#xff0c;谢谢大家支持 &#x1f496; 测试报告 1. 项目介绍2. 测试用例框架3. 自动化测试源码 1. 项目介绍 “星光日册”项目实现了用…

追踪攻击数据包中的真实IP地址:方法与技巧

在网络安全领域&#xff0c;追踪攻击数据包中的真实IP地址是一项至关重要的任务。通过确定攻击者的真实IP地址&#xff0c;可以有效地识别和阻止网络攻击行为&#xff0c;提高网络安全防御水平。IP数据云IP地址查询将介绍几种常用的方法和技巧&#xff0c;帮助安全人员有效追踪…

【Linux】CAN根据时钟频率、波特率计算采样点详解

1、采样点知识回顾 参考博客:【CAN】知识点:帧类型、数据帧结构、传输速率、位时间、采样点 CAN 采样点是指在一个数据位的传输周期内,接收器实际采样数据的时间点。这个时间点是以百分比来表示的,它决定了在数据位的传输周期中,何时读取数据位的值。 正确设置采样点对…

60*13薪,外包到新疆...去吗?

大家好&#xff0c;我是白露呀。 今天我在牛客上看到一篇帖子&#xff0c;一位网友说自己收到一个 offer &#xff0c;薪资很高&#xff1a;60k*13&#xff0c;大约一年有近80万。 但是有个要求是外包到新疆的乌鲁木齐&#xff0c;他拿不定主意&#xff0c;就在牛客上发了这个…

华为数据之道第三部分导读

目录 导读 第三部分 第7章 打造“数字孪生”的数据全量感知能力 “全量、无接触”的数据感知能力框架 数据感知能力的需求起源&#xff1a;数字孪生 数据感知能力架构 基于物理世界的“硬感知”能力 “硬感知”能力的分类 “硬感知”能力在华为的实践 基于数字世界的…
最新文章