Java ThreadPoolExecutor线程池有关介绍
作者:明天一定. 发布时间:2022-11-21 02:03:20
为什么要有线程池?
在实际使用中,服务器在创建和销毁线程上花费的时间和消耗的系统资源都相当大,所以要尽可能减少创建和销毁线程的次数。
由于没有线程创建和销毁时的消耗,可以提高系统响应速度
可以对线程进行合理的管理
线程池状态
1、RUNNING
状态说明:线程池处于RUNNING状态时,能够接收新任务以及对已添加的任务进行处理。
2、SHUTDOWN
状态说明:线程池处于SHUTDOWN状态时,不接收新任务,但能处理已添加的任务
3、STOP
状态说明:线程池处于STOP状态时,不接收新任务,不处理已添加的任务,并且会中断正在处理的任务
4、TIDYING
状态说明:当所有的任务已终止,ctl记录的任务数为0,线程池的状态会变为TIDYING状态;当线程池的状态变为TIDYING状态时,会调用钩子函数terminated(),该方法在ThreadPoolExecutor中是空的,若用户想在线程池变为TIDYING时进行相应的处理,就需要重载terminated()函数实现。
当线程池为STOP时,线程池中执行的任务为空时,就会又STOP->TIDYING
5、TERMINATED
状态说明:线程池彻底终止,就会变成TERMINATED状态
ThreadPoolExecutor核心参数
corePoolSize
corePoolSize – the number of threads to keep in the pool, even if they are idle, unless allowCoreThreadTimeOut is set
池中持有的线程数,即使它们处于空闲状态,除非设置了allowCoreThreadTimeOut
线程池中的核心线程数,当提交一个任务时,线程池创建一个新线程执行任务,直到当前线程数等于corePoolSize, 即使有其他空闲线程能够执行新来的任务, 也会继续创建线程;
如果执行了线程池的prestartAllCoreThreads()方法,线程池会提前创建并启动所有核心线程。
案例:
核心线程数和最大线程数为1,使用一个不存储元素的阻塞队列。
public static void main(String[] args) throws InterruptedException {
ThreadPoolExecutor executor = new ThreadPoolExecutor(1, 1, 0,
TimeUnit.SECONDS,
new SynchronousQueue<>(),
Executors.defaultThreadFactory(),
new ThreadPoolExecutor.AbortPolicy());
for (int i = 0; i < 2; i++) {
executor.execute(()->{
System.out.println(Thread.currentThread().getName());
});
Thread.sleep(1000);
}
}
输出 :
pool-1-thread-1
pool-1-thread-1
maximumPoolSize
maximumPoolSize – the maximum number of threads to allow in the pool
池中允许存在的最大线程数
案例:
核心线程数是1,最大线程数为3,使用一个不存储元素的阻塞队列。(注意结合workQueue参数食用~)
public static void main(String[] args) throws InterruptedException {
ThreadPoolExecutor executor = new ThreadPoolExecutor(1, 3, 0,
TimeUnit.SECONDS,
new SynchronousQueue<>(),
Executors.defaultThreadFactory(),
new ThreadPoolExecutor.AbortPolicy());
for (int i = 0; i < 3; i++) {
executor.execute(()->{
System.out.println(Thread.currentThread().getName());
});
}
}
输出:
pool-1-thread-1
pool-1-thread-3
pool-1-thread-2
keepAliveTime
keepAliveTime – when the number of threads is greater than the core, this is the maximum time that excess idle threads will wait for new tasks before terminating.
当线程数大于核心时,这是多余空闲线程在终止前等待新任务的最长时间。
线程空闲时的存活时间,即当线程没有任务执行时,该线程继续存活的时间;默认情况下,该参数只在线程数大于corePoolSize时才有用, 超过这个时间的空闲线程将被终止;
unit
unit – the time unit for the keepAliveTime argument
keepAliveTime参数的单位
workQueue
workQueue – the queue to use for holding tasks before they are executed. This queue will hold only the Runnable tasks submitted by the execute method.
用来放置没有执行的任务,此队列将仅保存execute方法提交的可运行任务。
用来保存等待被执行的任务的阻塞队列。
如果当前线程数为corePoolSize,继续提交的任务被保存到阻塞队列中,等待被执行;
如果当前阻塞队列满了,且继续提交任务,则创建新的线程执行任务,前提是当前线程数小于maximumPoolSize;当阻塞队列是 * 队列, 则maximumPoolSize则不起作用, 因为无法提交至核心线程池的线程会一直持续地放入workQueue。
JDK提供以下队列:
ArrayBlockingQueue: 基于数组结构的有界阻塞队列,按FIFO排序任务;(常用)
LinkedBlockingQueue: 基于链表结构的阻塞队列,按FIFO排序任务,吞吐量通常要高于ArrayBlockingQueue;(常用)
SynchronousQueue: 一个不存储元素的阻塞队列,每个插入操作必须等到另一个线程调用移除操作,否则插入操作一直处于阻塞状态,吞吐量通常要高于LinkedBlockingQueue;(常用)
PriorityBlockingQueue: 具有优先级的 * 阻塞队列;
DelayQueue:一个使用优先级队列实现的 * 阻塞队列。
LinkedTransferQueue:一个由链表结构组成的 * 阻塞队列。
LinkedBlockingDeque:一个由链表结构组成的双向阻塞队列。
案例:
核心线程数是1,最大线程数为3,使用一个容量为1的队列。
public static void main(String[] args) throws InterruptedException {
ThreadPoolExecutor executor = new ThreadPoolExecutor(1, 3, 0,
TimeUnit.SECONDS,
new ArrayBlockingQueue<>(1),
Executors.defaultThreadFactory(),
new ThreadPoolExecutor.AbortPolicy());
for (int i = 0; i < 3; i++) {
executor.execute(()->{
System.out.println(Thread.currentThread().getName());
});
}
}
输出:
pool-1-thread-2
pool-1-thread-1
pool-1-thread-2
threadFactory
threadFactory – the factory to use when the executor creates a new thread
创建执行器创建线程的工厂
通过自定义的线程工厂可以给每个新建的线程设置一个具有识别度的线程名。默认为DefaultThreadFactory。
案例:
给线程起名字。我是用spring里的类。如不想引入过多依赖,可以自己仿照Executors.defaultThreadFactory()的代码写一个类更改namePrefix即可。
public static void main(String[] args){
CustomizableThreadFactory customizableThreadFactory = new CustomizableThreadFactory("mine-");//import org.springframework.scheduling.concurrent.CustomizableThreadFactory;
ThreadPoolExecutor executor = new ThreadPoolExecutor(1, 1, 0,
TimeUnit.SECONDS,
new SynchronousQueue<>(),
customizableThreadFactory,
new ThreadPoolExecutor.CallerRunsPolicy());
for (int i = 0; i < 1; i++) {
executor.execute(()->{
System.out.println(Thread.currentThread().getName());
});
}
}
输出:
mine-1
handler
handler – the handler to use when execution is blocked because the thread bounds and queue capacities are reached
达到线程边界和队列容量而阻止执行时使用的处理程序
线程池的饱和策略,当阻塞队列满了,且没有空闲的工作线程,如果继续提交任务,必须采取一种策略处理该任务,
线程池提供了4种策略:
AbortPolicy
: 直接抛出异常,默认策略;
CallerRunsPolicy
: 用调用者所在的线程来执行任务;
DiscardOldestPolicy
: 丢弃阻塞队列中靠最前的任务,并执行当前任务;
DiscardPolicy
: 直接丢弃任务;
案例:
以 CallerRunsPolicy为案例。核心和最大线程数为1。
public static void main(String[] args) throws InterruptedException {
ThreadPoolExecutor executor = new ThreadPoolExecutor(1, 1, 0,
TimeUnit.SECONDS,
new SynchronousQueue<>(),
Executors.defaultThreadFactory(),
new ThreadPoolExecutor.CallerRunsPolicy());
for (int i = 0; i < 3; i++) {
executor.execute(()->{
System.out.println(Thread.currentThread().getName());
});
}
}
输出:
main
main
pool-1-thread-1
关闭线程池的方式
shutdown:
修改线程池状态为SHUTDOWN
不再接收新提交的任务
中断线程池中空闲的线程
第3步只是中断了空闲的线程,但正在执行的任务以及线程池任务队列中的任务会继续执行完毕
shutdownNow:
修改线程池状态为
STOP
不再接收任务提交
尝试中断线程池中所有的线程(包括正在执行的线程)
返回正在等待执行的任务列表
List<Runnable>
为什么不推荐使用Executors去创建线程池
newFixedThreadPool和newSingleThreadExecutor: 阻塞队列为 * 队列,主要问题是堆积的请求处理队列可能会耗费非常大的内存,甚至OOM。newCachedThreadPool和newScheduledThreadPool: 线程数最大数是Integer.MAX_VALUE,可能会创建数量非常多的线程,甚至OOM。
来源:https://blog.csdn.net/wai_58934/article/details/126712972
猜你喜欢
- 一、委托1、什么是委托委托是面向对象的、类型安全的,是引用类型。使用delegate关键字进行定义。委托的本质就是一个类,继承自System
- 一个activity就好比一个网页,此文章讲解怎样创建一个activity并且实现跳转!一、学习创建Activity1、新建一个java类,
- Android开发过程中,特别是新开的项目,底部状态栏的切换使用的频率非常的高,主要的实现方式有:(1)、TabLayout + Fragm
- 目录前言项目环境1.线程池示例2.shutdown3.isShutdown4.isTerminated5.awaitTermination6
- 我就废话不多说了,大家还是直接看代码吧~ public static void main(String[] args) { &n
- int a = 5; int b = 30; Console.WriteLine(a^b);&n
- Http协议的重要性相信不用我多说了,HttpClient相比传统JDK自带的URLConnection,增加了易用性和灵活性(具体区别,日
- 本文实例为大家分享了Android Studio实现简易计算器App的具体代码,供大家参考,具体内容如下效果演示布局文件<?xml v
- 背景Java是一种流行的编程语言,验证码是一种常用的网络安全技术。Java发展至今,网上也出现了各种各样的验证码,本人初学Java,下面是我
- 需求: 使用IO流将指定目录下的若干个音频文件的高潮部分,进行剪切,并重新拼接成一首新的音频文件 思路(以两首歌为例):第一首歌有
- 两个简单的例子,代码实现如下:1、随机拆分一个整数public static List<Integer> randomList(
- 本文实例讲述了Android开发高级组件之自动完成文本框(AutoCompleteTextView)用法。分享给大家供大家参考,具体如下:通
- 当我们要实现丰富的图文混排效果的时候,我们一般会使用webview,这是一个功能十分强大的的控件,来看看官方的解释:A View that
- windows xp下配置JDK环境变量:1.安装JDK,安装过程中可以自定义安装目录等信息,例如我们选择安装目录为D:/java/jdk1
- 测试例:PageElement pe = new PageElement();pe.LoadDataFromJsonString("
- Spring Cloud 本地属性覆盖注:使用版本版本 spring cloud F SR2当前在项目中使用了Spring cloud 配置
- public Bitmap CopyBitmap(Bitmap source){ int depth =
- 序、前言emmmmm,首先这篇文章讲的不是用BinaryFormatter来进行结构体的二进制转换,说真的BinaryFormatter这个
- 1.前言初始化就是给变量一个初始值。 初始化的目的是为了让变量有值,防止使用时出现异常。在构造函数中,有一项重要功能就是对成员变量进行初始化
- 前文传送门:NioEventLoop处理IO事件执行任务队列继续回到NioEventLoop的run()方法:protected void