首页 星云 工具 资源 星选 资讯 热门工具
:

PDF转图片 完全免费 小红书视频下载 无水印 抖音视频下载 无水印 数字星空

Java线程池详解

编程知识
2024年08月20日 16:35

Java线程池详解

线程池解释

线程池采用了池化思想,能够有效的管理线程的生命周期,减少了每次获取资源的消耗,提高了资源的利用率。类似池化实现还有数据库连接池、HTTP连接池等

好处

  • 减少了线程创建和销毁的开销
  • 提高了响应速度
  • 使得线程更加方便管理

常见使用场景

  • 量大处理时间较短的任务:有效利用线程池减少线程创建销毁的消耗
  • 需要限制线程数量时:线程越多对资源的消耗越大,线程池可以设置最大线程数进行限制
  • 异步执行一批不需要立刻反馈结果的任务:异步执行任务,减少系统响应时间

线程池的引入:Executor框架

Executor框架为Java 5 引入,将传统线程的操控方法进行优化升级,使其针对不同 场景更加的灵活、管理更加的方便,其核心jar包为java.util.concurrent,简称JUC。此外,很重要的一点,该框架还避免了 this逃逸问题。

this逃逸:指在构造函数返回之前,其他线程持有了该对象的引用,此时引用该对象调用其方法时,可能会出现不好排查的异常。

Executor框架主要包括三个部分

  • 任务:Runnabale/Callable,可以被 ThreadPoolExecutorScheduledThreadPoolExecutor(继承ThreadPoolExecutor) 执行

  • 执行:通过实现Executor接口的子接口ExecutorService去构造相对来说比较完整的线程池执行系统

  • 返回值:线程池的优势之一,通过实现Futrure接口的FutureTask 类将异步执行的结果获取到

  • 主线程创建 Runnable 或者 Callable 的任务对象,然后把 实现的 Runnbale 或者 Callable 交给 ExecutorService 执行:ExecutorService.execute(Runnable command) 或者 把 Runnable对象或者 Callable 对象提交给 ExecutorService 执行(ExecutorService.submit(Runnable command)

  • 如果执行 ExecutorService.submit()ExecutorService 返回一个实现了 Future 接口的对象(submit() 会返回一个 FuturesTask对象,FutureTask 实现了 Runnable,可以创建FutureTask,然后直接交给 ExecutorService执行;execute 则会把异常打印出来)

  • 最后,主线程执行 FuturesTask.get() 方法等待任务执行完成。也可以通过 FutureTask.cancel(boolean mayInterruptIfRunning) 来取消此任务执行。

ThreadPoolExecutor为例,ThreadPoolExecutor继承了AbstractExecutorService(抽象函数)的submit方法,并且实现了AbstractExecutorService来自Executor接口的execute方法,因此在调用线程池的submit方法时就会通过ThreadPoolExecutorexecute将任务(Runnabale/Callable为核心的RunnableFuture对象)加到工作队列中(addWorker方法实现),返回值则通过RunnableFuture对象的形式返回。

简单来说就是主线程将Runnabale/Callable对象通过submit方法提交给线程池,线程池通过内部调度按照不同的策略执行多线程任务,然后通过返回的Futrure对象的get方法取出执行结果。

ThreadPoolExecutor 类

线程池实现类 ThreadPoolExecutorExecutor 框架最核心的类

主线程提交任务后

  • 第一步,判断核心线程池线程数是否已满,若未满则创建线程,若已满则进入第二步
  • 第二步,判断等待队列是否已满,若未满则加入队列,若已满则进入第三步
  • 第三步,判断最大线程数是否已满,若未满则创建线程,若已满则根据设定的拒绝策略处理
参数介绍:
  • corePoolSize:核心线程数,能第一时间处理的线程数
  • maximumPoolSize:最大线程数,当核心线程池以及队列都满了的时候,线程池就会扩充到最大线程数
  • workQueue: 当核心线程池已满时,队列可以用来暂存后边进来的任务
  • keepAliveTime:线程数量大于核心线程数量时,线程闲置时间超过此值的线程会被回收掉,直至缩到核心线程数
  • unitkeepAliveTime 参数的时间单位
  • threadFactory:线程工厂,executor 创建线程时用到
  • handler:拒绝策略线程池已达到最大线程数,此时队列也满了,再有任务进来就会触发拒绝策略
拒绝策略:
  • AbortPolicy:拒绝任务并抛异常RejectedExecutionException
  • CallerRunsPolicy:直接在调用线程中执行任务,影响性能
  • DiscardPolicy:拒绝任务,不报异常,不做任何处理
  • DiscradOldestPolicy:拒绝掉最早的任务,执行最新的

线程池相关的内容有很多,该文章重在介绍,若要深究其中的某个元素,可以通过参考源码的方式,这样可以最直观的看到内部组成以及各个元素之间如何合作解决线程池的实现问题。

From:https://www.cnblogs.com/zyyjgsj/p/18369916
本文地址: http://shuzixingkong.net/article/1273
0评论
提交 加载更多评论
其他文章 XSS 基本概念和原理介绍
XSS 基本概念和原理介绍 基本概念 跨站脚本攻击 XSS(Cross Site Scripting),为了不和层叠样式表 ( Cascading Style Sheets,CSS ) 的缩写混淆,故将跨站脚本攻击缩写为 XSS。 恶意攻击者 往 Web 页面里插入恶意 JavaScript 代码,
XSS 基本概念和原理介绍 XSS 基本概念和原理介绍 XSS 基本概念和原理介绍
LLM应用实战: 产业治理多标签分类
本期的干货就是分享关于如何基于LLM实现数量多、层级多的多标签分类的实战经验,各位读者可以参考借鉴。
LLM应用实战: 产业治理多标签分类 LLM应用实战: 产业治理多标签分类
使用python-slim镜像遇到无法使用PostgreSQL的问题
前言 之前不是把 DjangoStarter 的 docker 方案重新搞好了吗 一开始demo部署是使用 SQLite 数据库的,用着没问题,但很快切换到 PostgreSQL 的时候就遇到问题了… 报错 docker 启动之后,app 容器报错 django.core.exceptions.Im
动态规划方法论
动态规划 动态规划,英文:Dynamic Programming,简称DP,如果某一问题有很多重叠子问题,使用动态规划是最有效的。 所以动态规划中每一个状态一定是由上一个状态推导出来的,这一点就区分于贪心,贪心没有状态推导,而是从局部直接选最优的, 我在这里举一个典型的动态规划的问题——背包问题:
前端使用 Konva 实现可视化设计器(21)- 绘制图形(椭圆)
本章开始补充一些基础的图形绘制,比如绘制:直线、曲线、圆/椭形、矩形。这一章主要分享一下本示例是如何开始绘制一个图形的,并以绘制圆/椭形为实现目标。
前端使用 Konva 实现可视化设计器(21)- 绘制图形(椭圆) 前端使用 Konva 实现可视化设计器(21)- 绘制图形(椭圆) 前端使用 Konva 实现可视化设计器(21)- 绘制图形(椭圆)
一文讲清楚算法刷题-计算机专业新生必看
哈喽,大家好,我是Sunny,你也可以叫我萨宁,一个热爱分享编程知识的程序员。我的昵称是Sunny不要停,寓意是美好的晴朗日子不要停下来,希望大家都能每天开开心心的。我的频道主要分享编程知识,生活,大学计算机学科学习,考研经验。目前已经上岸某211计算机专业,有大学学习,考研相关的问题,欢迎关注我,
一文讲清楚算法刷题-计算机专业新生必看 一文讲清楚算法刷题-计算机专业新生必看 一文讲清楚算法刷题-计算机专业新生必看
.NET 智能组件完全开源
Daniel Roth在2024年3月20日发布了一篇文章: .NET 智能组件简介 – AI 驱动的 UI 控件。文章主要介绍了.NET Smart Components,这是一系列可以快速轻松地添加到.NET应用程序中的AI驱动的UI组件。这些组件旨在简化在现有软件中添加AI功能的过程,无需花费
Prometheus部署以及问题解决
Prometheus作用: Prometheus监控(Prometheus Monitoring)是一种开源的系统监控和警报工具。它最初由SoundCloud开发并于2012年发布,并在2016年加入了云原生计算基金会(CNCF)。Prometheus监控旨在收集、存储和查询各种指标数据,以帮助用户
Prometheus部署以及问题解决 Prometheus部署以及问题解决 Prometheus部署以及问题解决