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

JAVAEE——线程池

文章目录

  • 线程池的概念
    • 什么是线程池?
  • 标准库中的线程池
    • 线程池的创建
    • 工厂模式
    • 工厂模式的用途
    • 线程池涉及到的类有哪些
      • Executor接口
      • ExecutorService接口
      • Executors工厂类
      • AbstractExecutorService虚类
      • ThreadPoolExecutor普通类
        • ThreadPoolExecutor内部的实现4个拒绝策略
  • 线程池的实现
    • 实现线程池

线程池的概念

什么是线程池?

我们先来说一下什么是线程池,线程池这个名字大家应该能够有一定的了解,我们计算机中进程会用到池化技术,
在这里插入图片描述
所谓池化技术,它其实就跟蓄水池一样,也就是提前创建好一批线程,并且不让其销毁,等到需要他们去做任务的时候再将其唤醒,这样子就可以有效的提高性能,至于为什么可以提高性能呢?我们通过一个例子来说明一下。
比如说我们要去银行取钱,那么有两种方式第一种就是通过柜台取钱,第二种就是通过ATM机自动取钱。
在这里插入图片描述
在这里插入图片描述
这时候就有了两种应对方式,如果是柜台的话你需要签名,出示证件,初始银行卡,然后输入密码,确认额度等等。办理起来就会很麻烦而且还得等待上班的时候才可以,但是我们老百姓就想取个多的话一两千少的话几百块因此不希望这么麻烦,所以就可以去ATM机去,这时候ATM就很方便只需要输入密码即可。而线程池也是这个道理,我们会提前创造好一部分线程放入我们创造的一块内存空间中,等到需要的时候在调用这个线程池里的线程,这样子我们就可以大幅度的缩小时间的消耗非常的高效。

标准库中的线程池

线程池的创建

首先我们来说一下如何创建一个线程池呢?这里涉及到了几个知识点我们来说一下。

使用 Executors.newFixedThreadPool(10) 能创建出固定包含 10 个线程的线程池.
返回值类型为 ExecutorService
通过 ExecutorService.submit 可以注册一个任务到线程池中.

我们来写一段代码。

public class Main {public static void main(String[] args) {ExecutorService executorService=Executors.newFixedThreadPool(10);}
}

这样子我们就创建了一个含有十个线程的线程池,这时候有的同学就有疑问了,为什么这里我们创建线程池的时候涉及到了两个类,一般不都是一个类然后new一下吗?就像我们创建线程一样,这是为什么呢?这里涉及到了一个模式叫做工厂模式

工厂模式

什么是工厂模式呢?我们举个例子假设说我们去买个本子,我们想买有一百张纸的本子,这时候店家肯定不会像A4纸那样给我们一张一张的本子肯定是,生产了一百张纸然后将这一百张纸进行装订和包装从而交到我们手上。那么什么地方要用到工厂模式呢?

工厂模式的用途

我们应该都知道函数重载吧。那么如果说存在这样一种情景,就是你需要写一个除法算法但是两个除法算法的要求不一样第一个方法的要求是返回值必须得保留两位小数,第二个方法的要求是保留三位小数并且必须和第一个方法的名字一样用重载实现。这时候就会有人说了怎么会有这么离谱的要求,是的如果放在这里干说确实很离谱但是如果是类里面的构造方法呢?构造方法的方法名称一样唯一能构成重载的就是参数类型,可是会不会有一些场景即使是参数类型也完全一样呢?答案是肯定的并且由于这是构造方法导致我们的函数名称还不能发生改变,因此最好的解决办法是什么呢?那就是再创造一个类这个类里面包含了很多我们需要的方法,接下来我们想要创建哪种类型只需要从这个类里面进行完成就可以了。这个就是工厂模式也就是我们上面的那个代码的来源。

线程池涉及到的类有哪些

我们打开源代码来看一些东西

Executor接口

任务执行器(Executor)是一个接口,位于java.util.concurrent包下,它的作用主要是为我们提供任务与执行机制(包括线程使用和调度细节)之间的解耦。比如我们定义了一个任务,我们是通过线程池来执行该任务,还是直接创线程来执行该任务呢?通过Executor就能为任务提供不同的执行机制。执行器的实现方式各种各样,常见的包括同步执行器、一对一执行器、线程池执行器、串行执行器等等。这个接口内部只有一个方法。如下图
在这里插入图片描述

ExecutorService接口

ExecutorService 主要用来管理和控制线程,是Java并发编程的重要工具,在实际业务中, ExecutorService 的使用场景非常广泛,比如,经常需要处理大量的用户请求,而每个请求都需要一个独立的线程去处理,如果直接为每个请求创建一个新的线程,会消耗大量的系统资源,而且管理起来也非常麻烦,此时,就可以使用ExecutorService来解决问题。这时候我们有个问题这个东西是个接口啊,接口是不能实例化处对象的,那么为什么刚刚那个工厂类中的创建线程池方法可以直接返回给这个接口呢?其实那个方法里所创造的对象根本不是这个接口的实例化,而是它的子类。因此这里其实涉及到了一个向上转型。

Executors工厂类

Executors是Java中用于创建线程池的工厂类,它提供了一系列的静态工厂方法,用于创建不同类型的线程池。这些工厂方法隐藏了线程池的复杂性,使得线程池的创建变得非常简单。Executors工厂类提供的线程

  • newCachedThreadPool():创建一个可缓存的线程池。这个线程池的线程数量可以根据需要自动扩展,如果有可用的空闲线程,就会重用它们;如果没有可用的线程,就会创建一个新线程。适用于执行大量的短期异步任务。
  • newSingleThreadExecutor():创建一个单线程的线程池。这个线程池中只包含一个线程,用于串行执行任务。适用于需要按顺序执行任务的场景。
  • newFixedThreadPool(int nThreads):创建一个固定大小的线程池,其中包含指定数量的线程。线程数量是固定的,不会自动扩展。适用于执行固定数量的长期任务。
  • newScheduledThreadPool(int corePoolSize):创建一个固定大小的线程池,用于定时执行任务。线程数量固定,不会自动扩展。适用于定时执行任务的场景。
  • newWorkStealingPool(int parallelism):创建一个工作窃取线程池,线程数量根据CPU核心数动态调整。适用于CPU密集型的任务。
  • newSingleThreadScheduledExecutor():创建一个单线程的定时执行线程池。只包含一个线程,用于串行定时执行任务。

AbstractExecutorService虚类

AbstractExecutorService实现了ExecutorService和Executor接口的基本方法,ThreadPoolExecute和ForkJoinPool继承AbstractExecutorService就可以减少实现的复杂度,接口适配器模式

ThreadPoolExecutor普通类

他是线程池的真正的实现类内部有很多实现细节

ThreadPoolExecutor内部的实现4个拒绝策略

我们保存任务

CallerRunsPolicy:由调用execute方法提交任务的线程来执行这个任务
AbortPolicy:抛出异常RejectedExecutionException拒绝提交任务
DiscardPolicy:直接抛弃任务,不做任何处理
DiscardOldestPolicy:去除任务队列中的第一个任务,重新提交

线程池的实现

实现线程池

import java.util.concurrent.ArrayBlockingQueue;
import java.util.concurrent.BlockingQueue;public class threadPool {private BlockingQueue<Runnable>blockingQueue=new ArrayBlockingQueue<>(10000);public void submit(Runnable runnable){try {blockingQueue.put(runnable);} catch (InterruptedException e) {throw new RuntimeException(e);}}public threadPool(int n){for(int i=0;i<4;i++){Thread thread=new Thread(()->{while(true){try {Runnable runnable=blockingQueue.take();runnable.run();} catch (InterruptedException e) {throw new RuntimeException(e);}}});thread.start();}}}
import java.beans.ExceptionListener;
import java.util.Objects;
import java.util.concurrent.Executor;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.ThreadPoolExecutor;public class Main {public static void main(String[] args) {threadPool th=new threadPool(100);for(int i=0;i<1000;i++){int id=i;th.submit(new Runnable() {@Overridepublic void run() {System.out.println("I love lele"+id);}});}}
}

那么以上代码就是我们创建线程池的代码这个代码结构相对是比较简单的。一个阻塞队列用来存储执行的任务然后创建线程去执行任务因为阻塞队列是线程安全的,因此当无任务的时候创建的线程会陷入等待状态中。

相关文章:

  • 利用R语言和curl库实现网页爬虫的技术要点解析
  • Unity AI Navigation自动寻路
  • Ruoyi-Cloud-Plus_使用Docker部署分布式微服务系统---SpringCloud工作笔记200
  • I/O模型的一些理解
  • Github万星项目lobe-chat,连接GPT4GPTs,平替chatgpt-plus
  • 探索 2024 年 Web 开发最佳前端框架
  • uniapp小程序路由跳转,使用uni.navigateBack方法,实现页面返回上一页
  • 蓝桥杯真题Day40 倒计时19天 纯练题!
  • vue表单rules校验是动态的
  • Notepad++:格式化json字符串(带转义)
  • Kubernetes概念:存储:PersistentVolume和PersistentVolumeClaim使用
  • C#WPF控件TextBlock、Label以及与TextBox的区别
  • Superset二次开发之webpack.config.js 功能模块解读
  • 目标检测的相关模型图:YOLO系列和RCNN系列
  • 城市内涝排水系统最新模型:慧天【HTWATER】与SWMM完美耦合
  • [iOS]Core Data浅析一 -- 启用Core Data
  • 【刷算法】求1+2+3+...+n
  • chrome扩展demo1-小时钟
  • CODING 缺陷管理功能正式开始公测
  • css属性的继承、初识值、计算值、当前值、应用值
  • Django 博客开发教程 16 - 统计文章阅读量
  • django开发-定时任务的使用
  • github指令
  • java8-模拟hadoop
  • Javascript基础之Array数组API
  • linux安装openssl、swoole等扩展的具体步骤
  • Linux后台研发超实用命令总结
  • SpiderData 2019年2月16日 DApp数据排行榜
  • SQL 难点解决:记录的引用
  • 高度不固定时垂直居中
  • 警报:线上事故之CountDownLatch的威力
  • 如何使用 JavaScript 解析 URL
  • 设计模式(12)迭代器模式(讲解+应用)
  • 通信类
  • 我感觉这是史上最牛的防sql注入方法类
  • 学习笔记:对象,原型和继承(1)
  • 正则表达式
  • 2017年360最后一道编程题
  • 如何用纯 CSS 创作一个货车 loader
  • ​LeetCode解法汇总2808. 使循环数组所有元素相等的最少秒数
  • ###51单片机学习(1)-----单片机烧录软件的使用,以及如何建立一个工程项目
  • #微信小程序(布局、渲染层基础知识)
  • (C#)获取字符编码的类
  • (二)学习JVM —— 垃圾回收机制
  • (实战)静默dbca安装创建数据库 --参数说明+举例
  • (原)本想说脏话,奈何已放下
  • (原創) 如何安裝Linux版本的Quartus II? (SOC) (Quartus II) (Linux) (RedHat) (VirtualBox)
  • (转)人的集合论——移山之道
  • (最完美)小米手机6X的Usb调试模式在哪里打开的流程
  • ******之网络***——物理***
  • .NET CORE 3.1 集成JWT鉴权和授权2
  • .NET Core工程编译事件$(TargetDir)变量为空引发的思考
  • .netcore 6.0/7.0项目迁移至.netcore 8.0 注意事项
  • [100天算法】-每个元音包含偶数次的最长子字符串(day 53)
  • [2019.3.5]BZOJ1934 [Shoi2007]Vote 善意的投票