当前位置: 首页 > news >正文

#详细介绍!!!线程池

本篇详细:

1.介绍了什么是线程池

2.使用线程池有什么好处

3.线程池的工作流程

4.线程池的各个参数介绍

5.如何编写Java代码来创建线程池

6.使用线程池的注意事项

目录

一:什么是线程池

二:为什么使用线程池来管理线程

三:线程池的工作流程

四:线程池的各个参数介绍

五:使用Java代码创建线程池

5.1 依靠Executors工厂类来创建

  5.2 依靠ThreadPoolExecutor类来创建

六:使用线程池的注意事项

1.合理的设置线程的数量



一:什么是线程池

线程池总的来说:就是创建的一个进行线程管理的容器集合,主要为了提高管理线程效率,是一种多线程处理的形式。

线程池:就是开发者根据需求创建多个线程放到一个池中(逻辑意义上),来统一管理这些线程,由池中的任务队列来直接给池中的空闲线程分配任务,大致为这样的一个过程

二:为什么使用线程池来管理线程

使用线程池的主要目的就是提高线程的调度执行和管理上的效率

优势:

1.降低资源的开销

没有线程池时线程的创建和销毁一般是由操作系统来进行的,而操作系统对线程的控制一般又是不可控的,那么当线程超过空闲时间,而线程就可能会被自动销毁,此时这中反复的操作就有可能出现线程频繁创建和销毁的情况,虽然说线程的创建和销毁的效率对比进程已经优化了很多,但是依旧会造成大量的系统资源上的浪费。

那么如果利用线程池来集中创建线程进行管理,并且核心线程创建完成后,没任务也不会进行销毁。此时就大大减少了系统频繁对线程的创建和销毁造成的资源浪费了。

2.提高任务的响应速度

任务需要线程来执行,而线程由系统进行调度又是无序的,那么就有可能在任务过多时,导致系统响应该任务过慢,此时这种把任务直接让系统来分配线程执行,我们称为用户态+内核态。

当我们(用户)把一个任务交给系统来分配线程(内核)来进行执行时,此时系统不一定会直接响应去执行我们的任务,因为系统需要同时处理大量的响应,导致任务获得响应的速度较低。

如果使用我们自己创建的线程池来进行任务的响应,那么此时就是一个纯用户态的操作了。

因为线程池中有自己的线程,去执行线程池对应的任务队列中的任务,当线程空闲就直接去队列中获取其他任务,直接响应;且由于有核心线程的存在,此时线程池中的核心线程一直都在,并不会被销毁,此时也省去了创建线程进行的时间开销,进一步使得响应速度变快。

3.可管理性

利用线程池集中统一管理,监控池中的线程,避免由于线程数量过多且线程之间相互进行抢占系统资源导致线程大面积堵塞

三:线程池的工作流程

大致如下图:

1.线程池在创建时,就会指定好各个参数的数量(例如:核心线程,临时线程等)

2.线程池有一个与之对应的任务队列(由于线程池作用的环境一般是多线程并发的状态,此时任务队列也是线程安全的数据结构)

3.我们在使用线程池时只需要往线程池的任务队列中添加任务,此时

        3.1如果池中的线程小于核心线程数就会创建核心线程再执行任务,且执行完不进行销毁

        3.2如果池中的线程大于核心线程数,小于临时线程数,此时就会把任务分配给空闲的线程如果线程全部都被占用此时就会创建临时线程来执行任务

        3.3如果池中线程数已满(核心线程数+临时线程数),此时就会运行线程池的拒绝策略

注意:线程池的拒绝策略后面单独进行讲解(重点面试题)

四:线程池的各个参数介绍

查看官方文档:

查看官方文挡,线程池的构造方法,我们可以看到如下的参数

参数介绍:

1. int corePoolSize :核心线程数

线程池中的核心线程创建之后就不会被销毁,需要用的时候,直接安排任务

2. int maximumPoolSize:最大线程数

最大线程数就是线程池允许同时存在的线程的最大数量:核心线程数+临时线程数

临时线程在执行完任务之后,空闲一定的时间会自动销毁,需要用的时候再创建

3.long keepAliveTime临时线程的空闲存活时间

与第2个对应,临时线程空闲一定时间后会自动销毁,而keepAliveTime就是设置的空闲时间

4.TimeUnit unit:时间单位

时间参数的单位:例如TimeUnit SECOND(秒),设置时间单位为秒

5.BlockingQueue<Runnable> workQueue:堵塞队列(用于记录存储需要执行的任务)

workQueue就是我们前面一直说的任务队列,线程池中都有一个用于记录任务的队列

BlockingQueue 为一个线程安全的阻塞队列,参数为Runnable:为一个实际任务,通过重写重写run方法来描述具体的任务逻辑

6.ThreadFactory threadFactory:工厂类:用于辅助线程创建的类(参考工厂模式)

7.RejectedExecutionHandler handler:线程池的拒绝策略

指定线程池的拒绝策略,如果线程池满了,依旧在添加任务,那么就可根据设置的拒绝策略来进行拒绝(专门讲线程池拒绝策略的时候再详细介绍)

五:使用Java代码创建线程池

5.1 依靠Executors工厂类来创建

1. newFixedThreadPool(int n),返回类型为ExecutorService接口类型

指定线程池中线程的数量为n,并且这些线程都为核心线程

ExecutorService pool = Executors.newFixedThreadPool(10);

2.newSingleThreadPool(),返回类型为ExecutorService接口类型

单例线程池,固定线程池中的线程数量为1,且为核心线程

ExecutorService pool = Executors.newSingleThreadExecutor();

3.newCatchThreadPool(),返回类型为ExecutorService接口类型

创建的线程池中核心线程数为0,最大线程数为Integer.MAX_VALUE

keepAliveTime:60 ; TimeUnit为SECOND(秒):临时线程空闲等待时间为60秒,时间到了自动销毁

总结:以这种方式创建的线程池,池中的线程都是临时线程

ExecutorService pool = Executors.newCachedThreadPool();

4.newScheduledThreadPool(int n),返回类型为ScheduledExecutorSerives接口类型

定时任务线程池:其中可设置池中线程定时或者周期性的执行池中的任务

n为核心线程数

介绍几种该线程池的安排任务的方法区别

4.1 调用schedule方法个线程池安排任务(普通定时任务)

4.2 调用scheduleAtFixedRate方法给线程池安排周期性任务(从任务开始时计算周期定时时间)

4.3调用scheduleWithFixedDelay方法给线程池安排周期性任务(从任务结束开始计算周期定时时间)

//1.普通定时任务
pool.scheduleAtFixedRate(new Runnable() {@Overridepublic void run() {System.out.println("hello1");}
},2,TimeUnit.SECONDS);//2.周期任务
pool.scheduleAtFixedRate(new Runnable() {@Overridepublic void run() {System.out.println("hello1");}
},2,5,TimeUnit.SECONDS);//3.周期任务
pool.scheduleWithFixedDelay(new Runnable() {@Overridepublic void run() {System.out.println("hello2");}
},2,5,TimeUnit.SECONDS);

上诉三个代码的作用:

1. 安排了一个打印hello1的任务,定时2秒后执行任务

2.安排了一个周期性打印hello2的任务,定时2秒后执行,并且从任务开始执行时开始计算,5秒后再次执行该任务,以这种周期执行

3.安排了一个周期性打印hello3的任务,定时2秒后执行,并且从任务结束是开始计算周期间隔时间,间隔时间为5秒

5.newSingleThreadScheduleExecutor();返回类型为ScheduledExecutorSerives接口类型

单个线程的定时任务或者周期任务的线程池。池中只有一个线程,其他与4基本一致

ScheduledExecutorService pool = Executors.newSingleThreadScheduledExecutor();

  5.2 依靠ThreadPoolExecutor类来创建

通过ThreadPoolExecutor类来更加灵活的创建线程池,根据ThreadPoolExecutor类的构造方法传递一系列参数进行灵活创建。

代码:

BlockingQueue<Runnable> queue = new LinkedBlockingQueue<>();ThreadPoolExecutor pool = new ThreadPoolExecutor(5,10,60,TimeUnit.SECONDS,queue);

指定一个线程安全的阻塞队列来存储线程池中的任务,

new ThreadPoolExecutor(5,10,60,TimeUnit.SECONDS,queue):

表示corePoolSize(核心线程数)为5;maximumPoolSize(最大线程数)为10

keepAliveTime(临时线程最大空闲时间)为60;TimeUnit.SECOND:单位为秒

queue为任务队列

当然还可以根据需求增加其他参数,参数详情参照前面线程池参数介绍

六:使用线程池的注意事项

1.合理的设置线程的数量

线程池的工作线程数设置应根据实际情况配置,CPU密集型业务(搜索、排序等)CPU空闲时间较少,线程数就不需要设置太多。N核服务器,通过执行业务的单线程分析出本地计算时间为x,等待时间为y,则工作线程数(线程池线程数)设置为 N*(x+y)/x,能让CPU的利用率最大化

相关文章:

#详细介绍!!!线程池

本篇详细&#xff1a; 1.介绍了什么是线程池 2.使用线程池有什么好处 3.线程池的工作流程 4.线程池的各个参数介绍 5.如何编写Java代码来创建线程池 6.使用线程池的注意事项 目录 一&#xff1a;什么是线程池 二&#xff1a;为什么使用线程池来管理线程 三&#xff1a;线程池…...

【嵌入式Linux学习笔记】基于Linux官方库的标准外设驱动

对于标准的外设如LED&#xff0c;KEY&#xff0c;PWM等&#xff0c;以及标准通信协议&#xff0c;Linux都自带有标准的驱动库&#xff0c;不需要我们自行编写&#xff0c;只需要配置好相应的GPIO属性和电气属性&#xff0c;即可匹配相应的驱动&#xff0c;在应用程序中直接使用…...

网络爬虫抓包工具

&#x1f4da;介绍&#xff1a;Charles是著名的抓包工具&#x1f402;&#xff0c;可以抓取移动端与pc端网络访问&#x1f577;的所有数据。我们将使用它抓取我们与小程序交互的所有信息。&#x1f387;我们可以百度搜索Charles官网下载适用于自己系统的Charles安装包&#x1f…...

蓝桥杯倒计时 | 倒计时17天

作者&#x1f575;️‍♂️&#xff1a;让机器理解语言か 专栏&#x1f387;&#xff1a;蓝桥杯倒计时冲刺 描述&#x1f3a8;&#xff1a;蓝桥杯冲刺阶段&#xff0c;一定要沉住气&#xff0c;一步一个脚印&#xff0c;胜利就在前方&#xff01; 寄语&#x1f493;&#xff1a…...

【Spring Cloud Alibaba】7.Sentinel熔断器仪表盘监控

文章目录简介什么是 Sentinel控制台获取源码方式下载jar包方式启动访问服务配置项目&#xff0c;启用Sentinel完整配置测试简介 接下来我们通过Sentinel控制台来实现对服务消费者提供的熔断机制进行监控和控制&#xff0c;本操作先要完成之前的步骤&#xff0c;详情请参照【Sp…...

个人博客系统项目测试报告

项目背景介绍 背景&#xff1a;当在学习一项技能的时候&#xff0c;我们总会习惯通过博客来记录所学的知识点&#xff0c;方便后期遗忘时随时查看和快速复习。本次开发的Web网站程序便是为了更加轻量和方便地记录自己的学习笔记 概述&#xff1a;一个Web网站程序&#xff0c;…...

flutter安装自用笔记

参照文章&#xff1a; 开发环境搭建 Flutter环境配置步骤&#xff1a; 1.系统配置要求 2.Java环境 3.Flutter SDK 4.Android 开发环境一、系统配置要求 操作系统&#xff1a;Windows 7 SP1 或更高的版本&#xff08;基于 x86-64 的 64 位操作系统&#xff09; 磁盘空间&…...

tomcat线程池以及在SpringBoot中的启动过程

tomcat两大组件&#xff1a;连接器Connector&#xff0c;容器Container tomcat线程池 Tomcat线程池扩展了ThreadPoolExecutor&#xff0c;行为稍有不同 重写了ThreadPoolExecutor的execute方法 如果总线程数达到maximumPoolSize&#xff0c;不会立刻抛RejectedExecutionExcept…...

第十四届中国大学生创新创业大赛

文章目录比赛官网比赛题目含金量非常高建议参加的学生推荐几个我感兴趣的题目联系比赛官网 官网地址&#xff1a;http://www.fwwb.org.cn/ 实际叫做&#xff1a;中国大学生创新创业大赛 比赛题目 题目公布查看地址&#xff1a;http://www.fwwb.org.cn/topic/index 题目有…...

LeetCode:322. 零钱兑换——动态规划从案例入门

&#x1f34e;道阻且长&#xff0c;行则将至。&#x1f353; &#x1f33b;算法&#xff0c;不如说它是一种思考方式&#x1f340;算法专栏&#xff1a; &#x1f449;&#x1f3fb;123 一、&#x1f331;322. 零钱兑换 题目描述&#xff1a;给你一个整数数组coins&#xff0c;…...

【lwIP(第四章)】网络接口

目录一、lwIP网络接口简介二、lwIP的netif结构三、lwIP的netif相关函数1. lwIP网络接口的全局变量2. netif_add()函数3. netif_remove()函数4. netif_set_default()函数一、lwIP网络接口简介 lwIP协议栈支持多种不同的网络接口&#xff08;网卡&#xff09;&#xff0c;由于网卡…...

Vue3 pinia入门篇(一)

系列文章目录 主要为了记录如何使用Pinia在Vue3中的使用方式&#xff08;下面会介绍为什么使用Vue3选型&#xff09; 文章目录系列文章目录不用Vue2使用Pinia举例子&#xff1f;1.笔者的个人看法&#xff1a;2.总结一、Pinia是什么1.状态管理工具&#xff08;类比Vuex&#xff…...

python面向对象编程解释

python是一个面向对象的编程语言 面向过程的开发语言有C&#xff0c;面向对象除了python还有java等语言 具体来讲&#xff1a; 面向过程 &#xff1a;举个例子&#xff0c;比如说&#xff0c;把大象装进冰箱总共分几步&#xff0c;第一步&#xff0c;把冰箱门打开&#xff0c…...

ARM(IMX6U)嵌入式软件裸机开发之环境搭建与配置

目录 前沿 Ubuntu 和 Windows 文件互传 Ubuntu 下 NFS 和 SSH 服务开启 Ubuntu 交叉编译工具链安装 Source Insight 软件安装和使用 Visual Studio Code 软件的安装和使用 前沿 为什么我们要学习裸机开发呢&#xff1f; 1、裸机开发是了解所使用的 CPU 最直接、最简单的方…...

Java文件复制多种方法

1、InputStream与OutputStream 创建两个文件 - 源和目标。然后我们从源创建InputStream并使用OutputStream将其写入目标文件进行 java 复制文件操作。 private static void copyFileUsingStream(File source, File dest) throws IOException {InputStream is null;OutputStr…...

Java语言-----封装、继承、抽象、多态、接口

目录 前言 一.封装 1.1封装的定义 1.2访问修饰符的使用 二.继承 2.1继承的定义 2.2继承的方法 2.3继承使用注意点 三.多态 3,1多态的定义 3.2动态绑定 3.3方法重写 3.4向上&#xff08;向下&#xff09;转型 四.抽象 4.1抽象的概述和定义 4.2抽象的使用 五…...

基于深度学习的瓶子检测软件(UI界面+YOLOv5+训练数据集)

摘要&#xff1a;基于深度学习的瓶子检测软件用于自动化瓶子检测与识别&#xff0c;对于各种场景下的塑料瓶、玻璃瓶等进行检测并计数&#xff0c;辅助计算机瓶子生产回收等工序。本文详细介绍深度学习的瓶子检测软件&#xff0c;在介绍算法原理的同时&#xff0c;给出Python的…...

仿网易云小程序(一)

目录 一、项目准备 二、项目初始化 1.新建项目 2.封装service请求 三、底部导航栏的设计 四、MV页面的设计 1.将获取到的数据进行渲染 2.播放量数据进行处理转换 3.时长数据进行处理转换 五、MV组件的抽离封装 六、请求的抽离video 七、下拉重新请求新的数据 八、跳转到…...

【C++】vector模拟实现及其应用

文章目录vector的介绍vector的使用及其实现vector的定义vector iterator 的使用vector空间增长问题vector的增删查改vector的介绍 vector是表示可变大小数组的序列容器。就像数组一样&#xff0c;vector也采用的连续存储空间来存储元素。也就是意味着可以采用下标对vector的元素…...

JS看这一篇就够啦,JS基础大全,可用于快速回顾知识,面试首选

1 JS简介 更多JS内容可以看MDN&#xff1a;点击传送 浏览器分成两部分&#xff1a;渲染引擎和 JS 引擎 渲染引擎&#xff1a;用来解析HTML与CSS&#xff0c;俗称内核&#xff0c;比如 chrome 浏览器的 blink &#xff0c;老版本的 webkitJS 引擎&#xff1a;也称为 JS 解释器…...

国防科技大学计算机基础课程笔记02信息编码

1.机内码和国标码 国标码就是我们非常熟悉的这个GB2312,但是因为都是16进制&#xff0c;因此这个了16进制的数据既可以翻译成为这个机器码&#xff0c;也可以翻译成为这个国标码&#xff0c;所以这个时候很容易会出现这个歧义的情况&#xff1b; 因此&#xff0c;我们的这个国…...

全球首个30米分辨率湿地数据集(2000—2022)

数据简介 今天我们分享的数据是全球30米分辨率湿地数据集&#xff0c;包含8种湿地亚类&#xff0c;该数据以0.5X0.5的瓦片存储&#xff0c;我们整理了所有属于中国的瓦片名称与其对应省份&#xff0c;方便大家研究使用。 该数据集作为全球首个30米分辨率、覆盖2000–2022年时间…...

大模型多显卡多服务器并行计算方法与实践指南

一、分布式训练概述 大规模语言模型的训练通常需要分布式计算技术,以解决单机资源不足的问题。分布式训练主要分为两种模式: 数据并行:将数据分片到不同设备,每个设备拥有完整的模型副本 模型并行:将模型分割到不同设备,每个设备处理部分模型计算 现代大模型训练通常结合…...

GruntJS-前端自动化任务运行器从入门到实战

Grunt 完全指南&#xff1a;从入门到实战 一、Grunt 是什么&#xff1f; Grunt是一个基于 Node.js 的前端自动化任务运行器&#xff0c;主要用于自动化执行项目开发中重复性高的任务&#xff0c;例如文件压缩、代码编译、语法检查、单元测试、文件合并等。通过配置简洁的任务…...

【笔记】WSL 中 Rust 安装与测试完整记录

#工作记录 WSL 中 Rust 安装与测试完整记录 1. 运行环境 系统&#xff1a;Ubuntu 24.04 LTS (WSL2)架构&#xff1a;x86_64 (GNU/Linux)Rust 版本&#xff1a;rustc 1.87.0 (2025-05-09)Cargo 版本&#xff1a;cargo 1.87.0 (2025-05-06) 2. 安装 Rust 2.1 使用 Rust 官方安…...

给网站添加live2d看板娘

给网站添加live2d看板娘 参考文献&#xff1a; stevenjoezhang/live2d-widget: 把萌萌哒的看板娘抱回家 (ノ≧∇≦)ノ | Live2D widget for web platformEikanya/Live2d-model: Live2d model collectionzenghongtu/live2d-model-assets 前言 网站环境如下&#xff0c;文章也主…...

[特殊字符] 手撸 Redis 互斥锁那些坑

&#x1f4d6; 手撸 Redis 互斥锁那些坑 最近搞业务遇到高并发下同一个 key 的互斥操作&#xff0c;想实现分布式环境下的互斥锁。于是私下顺手手撸了个基于 Redis 的简单互斥锁&#xff0c;也顺便跟 Redisson 的 RLock 机制对比了下&#xff0c;记录一波&#xff0c;别踩我踩过…...

2025年低延迟业务DDoS防护全攻略:高可用架构与实战方案

一、延迟敏感行业面临的DDoS攻击新挑战 2025年&#xff0c;金融交易、实时竞技游戏、工业物联网等低延迟业务成为DDoS攻击的首要目标。攻击呈现三大特征&#xff1a; AI驱动的自适应攻击&#xff1a;攻击流量模拟真实用户行为&#xff0c;差异率低至0.5%&#xff0c;传统规则引…...

C# WPF 左右布局实现学习笔记(1)

开发流程视频&#xff1a; https://www.youtube.com/watch?vCkHyDYeImjY&ab_channelC%23DesignPro Git源码&#xff1a; GitHub - CSharpDesignPro/Page-Navigation-using-MVVM: WPF - Page Navigation using MVVM 1. 新建工程 新建WPF应用&#xff08;.NET Framework) 2.…...

C#最佳实践:为何优先使用as或is而非强制转换

C#最佳实践&#xff1a;为何优先使用as或is而非强制转换 在 C# 的编程世界里&#xff0c;类型转换是我们经常会遇到的操作。就像在现实生活中&#xff0c;我们可能需要把不同形状的物品重新整理归类一样&#xff0c;在代码里&#xff0c;我们也常常需要将一个数据类型转换为另…...