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

【JavaEE】线程池

目录

前言

什么是线程池

线程池的优点

ThreadPollExecutor中的构造方法

corePoolSize && maximumPoolSize

keepAliveTime && unit

workQueue

 threadFactory

如何在java中使用线程池

1.创建线程池对象

 2.调用submit添加任务

3.调用shutdown关闭线程池

手动实现线程池

创建一个线程池类

测试


前言

在前面我们都是通过new Thread() 来创建线程的,虽然在java中对线程的创建、中断、销毁、等值等功能提供了支持,但从操作系统角度来看,频繁的创建和销毁线程,是需要大量的时间和资源的。那么我们能不能先把线程提前从系统中申请好,需要使用线程的时候,直接从这个地方取出来,而不是从系统中重新申请,等线程用完之后,也是放回到这个地方,而不用频繁的进行创建和销毁呢?那我们就需要用到线程池,线程池能够提高程序的效率

什么是线程池

线程池 (ThreadPool) 是一种多线程处理形式,处理过程中将任务添加到队列,然后在创建后保持活动的线程中执行这些任务。线程池的主要目的是减少创建和销毁线程的开销,提高响应速度和整体性能。

为什么说线程池里取线程,比从系统中创建一个线程效率要高呢?

我们从线程池里直接取出线程是纯用户态操作,而我们新建一个线程是内核态+用户态配合完成的。

内核态:指操作系统内核及其相关组件(如设备驱动程序)运行的状态。

用户态:指用户程序(如应用程序和服务)运行的状态。

操作系统=操作系统内核+操作系统配套的应用程序

假设现在要去银行办银行卡,那么银行内部人员就是内核态,前往办卡的就为用户态。

如果我们想要办卡,就走到小窗口向柜员说明情况,此时柜员就会帮你进行操作。但如果此过柜员问你要身份证复印件,如果没有,那么就需要去进行打印。

若请求柜员帮你进行打印,那么柜员什么时候打印,这个是不确定的,效率较低。这种就是用户态+内核态,调用系统API,由系统内核来完成这一系列操作。

若我们自己去自助复印机进行复印,整个过程就是连贯可控的,效率较高。这种就是纯用户态,由我们自己控制。

线程池的优点

  • 减少资源消耗:通过重复利用已创建的线程,减少了创建和销毁线程所需的资源开销。
  • 提高响应速度:当有新的任务到达时,若线程池中有空闲的线程,任务可以直接执行,无需等待新的线程创建。
  • 防止资源耗尽:通过限制线程池中的最大线程数,可以避免因创建过多线程而引起的资源耗尽问题。
  • 提高线程利用率:线程池中的线程可以被重复利用。

线程池最大的好处就是减少每次启动、销毁线程的损耗。

ThreadPollExecutor中的构造方法

在java中,给我们提供了现成的线程池供我们来使用。  

我们可以看一下标准库中的线程池,在java.util.concurrent中给我们提供线程池的相关方法。

在点开concurrent之后,我们在接口中往下划,找到ThreadPoolExecutor类。我们可以看到,ThreadPoolExecutor类的构造方法有着好几个参数,那么这些参数都是什么意思呢?

‘’

 我们拿第四个构造方法来看。

corePoolSize && maximumPoolSize

corePoolSize:即核心线程数

maximumPoolSize:最大核心线程数

线程池可以支持“线程扩容”,若某个线程池初始状态下有m个线程,但m若不够用,就会自动增加m的个数。

在java标准库的线程池中,把线程分为两类:核心线程和非核心线程

核心线程可以理解为最少有多少个线程,而非核心线程就是线程扩容新增的线程。

核心线程数会始终存在线程池内部,而非核心线程,在繁忙的时候就会被创建出来,在空闲时就会把这些线程释放掉。

最大核心线程数=核心线程数+非核心线程数的最大值

keepAliveTime && unit

 那么非核心线程在空闲的时候就会被立即释放掉吗?

若在空闲一会后,又有新的任务,那岂不是要重新新增线程。

所以在构造方法中,给我们提供了两个参数,用来设置非核心线程在空闲时等待任务的最长时间。

keepAliveTime:表示线程池中空闲的非核心线程在终止前等待新任务的最长时间。

unit:用来设置keepAliveTime参数的时间单位。

workQueue

我们可以看到workQueue的类型为BlockingQueue<Runnable>,说明这是个用来存放任务的阻塞队列。可以根据需要设置阻塞队列的类型,如果需要优先级,则可以使用PriorityBlockingQueue;如果不需要优先级且任务的数目是恒定的,则可以使用ArrayBlockingQueue;如果任务的数目不是恒定的,则可以使用LinkedBlockingQueue

作用:

  1. 任务缓存:当线程池中的线程都在执行任务时,新提交的任务会被暂时存储在工作队列中,直到有空闲线程来执行它们。
  2. 任务调度:工作队列可以帮助线程池按照一定的策略来调度任务的执行顺序。
  3. 资源管理:通过限制工作队列的大小,可以控制线程池中等待执行的任务数量,从而避免资源耗尽的情况。

 threadFactory

 threadFactory:创建线程的⼯⼚,参与具体的创建线程⼯作.通过不同线程⼯⼚创建出的线程相当于 对⼀些属性进⾏了不同的初始化设置。

*handler(类型RejectedExecutionHandler)

最后一个参数是有关队列满后不要进行阻塞,而是要进行拒绝了,我们可以在构造方法上面查看线程池的拒绝策略有哪些.

在java提供了4种拒绝策略

  1. ThreadPoolExecutor.AbortPolicy:有新的任务想要入队时,直接抛出异常
  2. ThreadPoolExecutor.CallerRunsPolicy:新添加的任务线程池拒绝执行,由添加任务的线程执行该任务
  3. ThreadPoolExecutor.DiscardOldestPolicy:丢弃掉最旧的未被处理的请求,执行新的任务
  4. ThreadPoolExecutor.DiscardPolicy:丢弃掉当前新加的任务,不执行

讲完了线程池中的构造方法以及拒绝策略,那么如何在java中来使用线程池呢?

如何在java中使用线程池

1.创建线程池对象

         ExecutorService ex=Executors.newFixedThreadPool(4);

ExecutorService是一个继承于Executor的接口,主要用来管理和控制线程,是Java并发编程的重要工具。

在java标准库中,虽然ThreadPoolExecutor功能强大,但是使用起来比较麻烦,所以java标准库对这个类进行了封装:Executors类

 Executors类Java中用于创建线程池的工厂类,它提供了一系列的静态工厂方法,用于创建不同类型的线程池。

我们可以看到Executors工厂类中的工厂方法有以下几种类型:

  1. newCachedThreadPool()创建一个可缓存的线程池。这个线程池的线程数量可以根据需要自动扩展,如果有可用的空闲线程,就会重用它们;如果没有可用的线程,就会创建一个新线程。适用于执行大量的短期异步任务。
  2. newFixedThreadPool(int nThreads)创建一个固定大小的线程池,其中包含指定数量的线程。线程数量是固定的,不会自动扩展。适用于执行固定数量的长期任务。
  3. newSingleThreadExecutor()创建一个单线程的线程池。这个线程池中只包含一个线程,用于串行执行任务。适用于需要按顺序执行任务的场景。
  4. newScheduledThreadPool(int corePoolSize)创建一个固定大小的线程池,用于定时执行任务。线程数量固定,不会自动扩展此线程池;可以安排在给定延迟后运行命令或者定期执行。
  5. newSingleThreadScheduledExecutor()创建一个单线程的定时执行线程池。只包含一个线程,用于串行定时执行任务。
  6. newWorkStealingPool(int parallelism)创建一个工作窃取线程池,线程数量根据CPU核心数动态调整。这种线程池适合处理大量细粒度的可取消任务。

 2.调用submit添加任务

这里我们创建的是一个固定线程数为4的线程池,在线程池中,我们想要添加任务需要调用submit()方法,可以看到需要的参数是一个Runnable类型的任务。

 我们可以使用Lambda表达式或者直接new Runnable。(这里用第一个)

         for(int i=0;i<100;i++) {int id = i;ex.submit(() -> {System.out.println(Thread.currentThread().getName() + "  任务:" + id);});}

当我们运行之后,会发现任务都结束后,程序还没有停止,这是为什么呢?

在 Java 中,线程池创建的线程默认为前台线程,虽然main线程结束了,但是线程池里的前台线程仍然存在。

3.调用shutdown关闭线程池

如果我们想要关闭线程池中的线程,我们可以使用shutdown方法。

  public static void main(String[] args) {ExecutorService ex=Executors.newFixedThreadPool(4);for(int i=0;i<100;i++) {int id = i;ex.submit(() -> {System.out.println(Thread.currentThread().getName() + "  任务:" + id);});}ex.shutdown();}

既然我们学会如何使用java中的线程池,那么我们可以来实现一个线程池.

手动实现线程池

创建一个线程池类

由于这里是使用的阻塞队列,在队列满的时候若添加新的任务,会进入阻塞等待状态,不同于java线程池里的拒绝策略。

/*** 自定义线程池执行器类* 该类通过实现一个具有固定大小的线程池和一个阻塞队列来管理线程,用于异步执行任务*/
class MyThreadPoolExecutor {// 创建阻塞队列,用于存放待执行的任务// 队列大小设为1000,用于控制并发任务的数量,避免过多任务导致资源耗尽BlockingQueue<Runnable>  blockingQueue=new ArrayBlockingQueue<>(1000);/*** 构造函数,初始化线程池* 创建一个线程,该线程循环从阻塞队列中取任务并执行* 这个线程是线程池中的工作线程,负责执行提交的任务*/public MyThreadPoolExecutor(int n) {for (int i = 1; i <= n; i++) {Thread t = new Thread(() -> {// 无限循环,确保线程池可以持续处理任务,直到程序中断或阻塞队列被清空while (true) {try {// 从阻塞队列中取出一个任务,如果队列为空,则线程被阻塞,直到有任务放入队列Runnable task = blockingQueue.take();// 执行取出的任务task.run();} catch (InterruptedException e) {// 如果线程在等待状态时被中断,抛出运行时异常// 这通常会导致程序异常终止throw new RuntimeException(e);}}});// 启动线程池中的工作线程t.start();}}/*** 提交一个任务到线程池* @param task 需要被执行的任务*            任务被放入阻塞队列中,随后由线程池中的工作线程执行*/public void submit(Runnable task){// 将任务放入阻塞队列,如果队列已满,则操作会阻塞,直到有空间可用blockingQueue.offer(task);}
}

测试

class DemoTest1{public static void main(String[] args) throws InterruptedException {MyThreadPoolExecutor ex=new MyThreadPoolExecutor(4);for(int i=0;i<100;i++) {int id = i;ex.submit(()->{System.out.println(Thread.currentThread().getName()+"  任务:"+id);});}}
}


本篇就先到这了~
若有不足,欢迎指正~

相关文章:

【JavaEE】线程池

目录 前言 什么是线程池 线程池的优点 ThreadPollExecutor中的构造方法 corePoolSize && maximumPoolSize keepAliveTime && unit workQueue threadFactory 如何在java中使用线程池 1.创建线程池对象 2.调用submit添加任务 3.调用shutdown关闭线程池…...

lvs实战项目-dr模式实现

一、环境准备 主机名IP地址router eth0&#xff1a;172.25.254.100 eth1&#xff1a;192.168.0.100 clienteth0&#xff1a;172.25.254.200lvseth1&#xff1a;192.168.0.50web1web2 1、client配置 [rootclient ~]# cat /etc/NetworkManager/system-connections/eth0.nmconne…...

JSONP跨域

1 概述 定义 json存在的意义&#xff1a; 不同类型的语言&#xff0c;都能识别json JSONP(JSON with Padding)是JSON的一种“使用模式”&#xff0c;可用于解决主流浏览器的跨域数据访问的问题。由于同源策略&#xff0c;一般来说位于 server1.example.com 的网页无法与不是 s…...

Linux--shell脚本语言—/—终章

一、shell函数 1、shell函数定义格式 参数说明&#xff1a; 1、可以带function fun() 定义&#xff0c;也可以直接fun() 定义,不带任何参数。 2、参数返回&#xff0c;可以显示加&#xff1a;return 返回&#xff0c;如果不加&#xff0c;将以最后一条命令运行结果&#xff…...

免费代理池是什么,如何使用代理IP进行网络爬虫?

互联网是一个庞大的数据集合体&#xff0c;网络信息资源丰富且繁杂&#xff0c;想要从中找到自己需要的信息要花费较多的时间。为了解决这个问题&#xff0c;网络爬虫技术应运而生&#xff0c;它的主要作用就是在海量的互联网信息中进行爬取&#xff0c;抓取有效信息并存储。然…...

CAN直接网络管理(20240805)

长安CAN网络管理规范 个人理解&#xff1a;管理CAN网络中各NM节点的工作模式&#xff08;状态&#xff09;&#xff1b; 1.术语定义 &#x1f449;节点地址&#xff1a;用于唯一标识网络中每个节点的单字节数字&#xff0c;取值范围是 0x00~0xFF。&#x1f449;状态迁移&#x…...

HTML5+CSS3笔记(Xmind格式):第二天

Xmind鸟瞰图&#xff1a; 简单文字总结&#xff1a; 新增选择器&#xff1a; 1.选择相邻兄弟 2.属性选择器 3.结构性伪类选择器 4.整体结构类型 5.标签结构类型 6.指定子元素的序号 7.文本选择伪元素 8.表单中使用的状态伪类选择器 9.内容…...

视频压缩文件太大了怎么缩小?6个视频压缩技巧,速度收藏起来!

高清视频文件&#xff0c;尤其是那些以 1080p 和 720p 清晰度为特征的视频&#xff0c;通常都拥有相当大的体积&#xff0c;会占据大量计算机存储空间。因此&#xff0c;为了更好地将它们进行分享和存储&#xff0c;您可能需要对它们进行压缩&#xff0c;以减小它们的尺寸。然而…...

Python接口自动化测试数据提取分析:Jmespath

1、引言 在处理JSON数据时&#xff0c;我们常常需要提取、筛选或者变换数据。手动编写这些操作的代码不仅繁琐&#xff0c;而且容易出错。Python作为一个功能强大的编程语言&#xff0c;拥有丰富的库和工具来处理这些数据。今天&#xff0c;将介绍一个实用的Python库——JMESP…...

特种设备作业叉车司机题库及答案

1.在我们平时工作中&#xff0c;经常接触的汽油、柴油、机油、油棉纱、木材等均为() A、助燃物质 B、可燃物质 C、着火源 参考答案:B 2.叉车满载行驶时&#xff0c;如合成重心靠后() A、有利于纵向稳定 B、有利于横向稳定 C、纵向和横向均有利 参考答案:A 3.蓄电池车行驶中放…...

Linux 操作系统速通

一、安装虚拟机 1. VmWare 安装下载 vmware workstation pro 16 下载 win R 输入 ncpa.cpl 确保网卡正常 2. CentOS 系统下载 CentOS 系统下载 将 CentOS 系统安装到虚拟机 3. 查看虚拟机 IP 命令 ifconfig 4. finalShell 安装下载 finalShell 下载 输入用户名一般是 ro…...

IIS漏洞大全(附修复方法)

IIS6.0 IlS Server 在 Web 服务扩展中开启了 WebDAV&#xff0c;配置了可以写入的权限&#xff0c;造成任意文件上传。 漏洞复现 fofa:"llS-6.0" or 本地搭建2003 server 1)开启 WebDAV 和写权限: 做好准备工作后开启环境&#xff0c;然后我们去访问配置的IP&#…...

HarmonyOS笔记3:从网络数据接口API获取数据

面向HarmonyOS的移动应用一般采用MVVM模式&#xff08;见参考文献【1】&#xff09;&#xff0c;其中&#xff1a; M&#xff08;Model层)&#xff1a;模型层&#xff0c;存储数据和相关逻辑的模型。它表示组件或其他相关业务逻辑之间传输的数据。Model是对原始数据的进一步处理…...

Mac 下生成core dump

mac下生成core dump 使用ulimit -c查看ulimit设置,显示unlimited表示开启,显示0表示关闭,通过ulimit -c unlimited打开设置; 但是这个只在当前窗口有效果。如果需要变成系统全局设置。 就需要去改/etc/profile文件&#xff0c;打开&#xff0c;然后加上ulimit -c unlimited就可…...

详解Xilinx FPGA高速串行收发器GTX/GTP(1)--SerDes和GTX的关系

目录 1、SerDes和GTX的关系 2、传输总线的变化 2.1、从串行到并行 2.2、从并行又回到串行 文章总目录点这里:《FPGA接口与协议》专栏的说明与导航 1、SerDes和GTX的关系 Hold On,这个系列文章不是讲GTX收发器的吗?怎么一开始就扯到SerDes上了?GTX和SerDes之间有…...

golang实现Digest认证鉴权接口

什么是Digest认证鉴权接口? Digest认证鉴权接口是一种基于摘要算法的身份验证方法,用于确保API请求的安全性。在实际应用中,常常使用HTTP协议的Digest认证鉴权接口来验证请求的合法性。下面是一种常见的Digest认证鉴权流程: 1. 客户端发送HTTP请求到服务器,请求接口资源…...

机房托管服务器说明

机房托管服务器是指将企业或个人的服务器放置到专业数据中心(IDC机房)进行管理和维护&#xff0c;由数据中心提供稳定、安全的运行环境以及网络连接等基础设施支持。rak小编为您整理发布机房托管服务器说明详细内容。 通过托管服务器到专业机房&#xff0c;企业能够享受到高性能…...

CookieMaker工作室合作开发C++项目十一:拟态病毒

&#xff08;注&#xff1a;本文章使用了“无标题技术”&#xff09; 一天&#xff0c;我和几个同事&#xff0c;平台出了点BUG&#xff0c;居然给我刷出了千年杀&#xff0c;同事看得瑕疵欲裂&#xff0c;发誓要将我挫骨扬灰—— &#xff08;游戏入口&#xff1a;和平精英31.…...

57、PHP 实现 从扑克牌中随机抽取5张牌,判断是不是一个顺子

题目&#xff1a; PHP 实现 从扑克牌中随机抽取5张牌&#xff0c;判断是不是一个顺子 描述&#xff1a; 即这5张牌是不是连续的2-10位数字本身&#xff0c;A为1&#xff0c;J为11&#xff0c;Q为12&#xff0c;K为13&#xff0c;而大小王可以看成任意数字。 解题思路&#xf…...

前端HTML+CSS查漏补缺——仿制百度搜索首页的一些思考

在像素模仿百度搜索首页的时候&#xff0c;在实现的时候&#xff0c;遇到了一些值得记录的点。 在这个过程中&#xff0c;也顺便看了看百度的源码&#xff0c;感觉很有意思。 对了&#xff0c;QQ截屏里面获取到的颜色&#xff0c;是不大正确的&#xff0c;会有点误差。 这是我…...

XML Group端口详解

在XML数据映射过程中&#xff0c;经常需要对数据进行分组聚合操作。例如&#xff0c;当处理包含多个物料明细的XML文件时&#xff0c;可能需要将相同物料号的明细归为一组&#xff0c;或对相同物料号的数量进行求和计算。传统实现方式通常需要编写脚本代码&#xff0c;增加了开…...

8k长序列建模,蛋白质语言模型Prot42仅利用目标蛋白序列即可生成高亲和力结合剂

蛋白质结合剂&#xff08;如抗体、抑制肽&#xff09;在疾病诊断、成像分析及靶向药物递送等关键场景中发挥着不可替代的作用。传统上&#xff0c;高特异性蛋白质结合剂的开发高度依赖噬菌体展示、定向进化等实验技术&#xff0c;但这类方法普遍面临资源消耗巨大、研发周期冗长…...

高频面试之3Zookeeper

高频面试之3Zookeeper 文章目录 高频面试之3Zookeeper3.1 常用命令3.2 选举机制3.3 Zookeeper符合法则中哪两个&#xff1f;3.4 Zookeeper脑裂3.5 Zookeeper用来干嘛了 3.1 常用命令 ls、get、create、delete、deleteall3.2 选举机制 半数机制&#xff08;过半机制&#xff0…...

WEB3全栈开发——面试专业技能点P2智能合约开发(Solidity)

一、Solidity合约开发 下面是 Solidity 合约开发 的概念、代码示例及讲解&#xff0c;适合用作学习或写简历项目背景说明。 &#x1f9e0; 一、概念简介&#xff1a;Solidity 合约开发 Solidity 是一种专门为 以太坊&#xff08;Ethereum&#xff09;平台编写智能合约的高级编…...

微信小程序云开发平台MySQL的连接方式

注&#xff1a;微信小程序云开发平台指的是腾讯云开发 先给结论&#xff1a;微信小程序云开发平台的MySQL&#xff0c;无法通过获取数据库连接信息的方式进行连接&#xff0c;连接只能通过云开发的SDK连接&#xff0c;具体要参考官方文档&#xff1a; 为什么&#xff1f; 因为…...

k8s业务程序联调工具-KtConnect

概述 原理 工具作用是建立了一个从本地到集群的单向VPN&#xff0c;根据VPN原理&#xff0c;打通两个内网必然需要借助一个公共中继节点&#xff0c;ktconnect工具巧妙的利用k8s原生的portforward能力&#xff0c;简化了建立连接的过程&#xff0c;apiserver间接起到了中继节…...

tree 树组件大数据卡顿问题优化

问题背景 项目中有用到树组件用来做文件目录&#xff0c;但是由于这个树组件的节点越来越多&#xff0c;导致页面在滚动这个树组件的时候浏览器就很容易卡死。这种问题基本上都是因为dom节点太多&#xff0c;导致的浏览器卡顿&#xff0c;这里很明显就需要用到虚拟列表的技术&…...

技术栈RabbitMq的介绍和使用

目录 1. 什么是消息队列&#xff1f;2. 消息队列的优点3. RabbitMQ 消息队列概述4. RabbitMQ 安装5. Exchange 四种类型5.1 direct 精准匹配5.2 fanout 广播5.3 topic 正则匹配 6. RabbitMQ 队列模式6.1 简单队列模式6.2 工作队列模式6.3 发布/订阅模式6.4 路由模式6.5 主题模式…...

Yolov8 目标检测蒸馏学习记录

yolov8系列模型蒸馏基本流程&#xff0c;代码下载&#xff1a;这里本人提交了一个demo:djdll/Yolov8_Distillation: Yolov8轻量化_蒸馏代码实现 在轻量化模型设计中&#xff0c;**知识蒸馏&#xff08;Knowledge Distillation&#xff09;**被广泛应用&#xff0c;作为提升模型…...

iOS性能调优实战:借助克魔(KeyMob)与常用工具深度洞察App瓶颈

在日常iOS开发过程中&#xff0c;性能问题往往是最令人头疼的一类Bug。尤其是在App上线前的压测阶段或是处理用户反馈的高发期&#xff0c;开发者往往需要面对卡顿、崩溃、能耗异常、日志混乱等一系列问题。这些问题表面上看似偶发&#xff0c;但背后往往隐藏着系统资源调度不当…...