线程池ThreadPoolExecutor参数分析
概述
比如去火车站买票, 有7个(maximumPoolSize)售票窗口, 但只有3个(corePoolSize)窗口对外开放。那么对外开放的3个窗口称为核心线程数, 而最大线程数是7个窗口。
如果3个窗口都被占用, 那么后来的人就必须在售票厅(SynchronousQueue、LinkedBlockingQueue或者ArrayBlockingQueue)排队。
但后来售票厅人越来越多, 已经人满为患, 就类似于线程队列已满。这时候火车站站长下令, 把剩下的4个窗口也打开, 也就是目前已经有7个窗口同时运行。
后来又来了一批人,7个窗口也处理不过来了, 而且售票厅人已经满了, 这时候站长就下令封锁入口,不允许其他人再进来, 这就是线程拒绝策略。
线程存活时间指的是, 允许售票员休息的最长时间, 以此限制售票员偷懒的行为。
售票厅-SynchronousQueue
public static void main(String[] args) {
// 核心线程数
int corePoolSize = 3;
//最大线程数
int maximumPoolSize = 7;
// 非核心线程的最大空闲时间
long keepAliveTime = 10L;
// 队列种类
SynchronousQueue synchronousQueue = new SynchronousQueue<Runnable>();
//LinkedBlockingQueue linkedBlockingQueue = new LinkedBlockingQueue<Runnable>();
//ArrayBlockingQueue arrayBlockingQueue = new ArrayBlockingQueue<Runnable>(5);
ThreadPoolExecutor threadpool = new ThreadPoolExecutor(corePoolSize, maximumPoolSize, keepAliveTime, TimeUnit.SECONDS, synchronousQueue);
for (int i = 0; i < 20; i++) {
threadpool.execute(() -> {
System.out.println("线程名称:" + Thread.currentThread().getName());
try {
Thread.sleep(50000L);
} catch (Exception ex) {
ex.printStackTrace();
}
});
System.out.println("当前线程数量:" + threadpool.getPoolSize());
System.out.println("当前队列大小:" + synchronousQueue.size());
}
}
输出结果
当前线程数量:1
Exception in thread "main" java.util.concurrent.RejectedExecutionException: Task org.tonny.threads.threadpool.ThreadPool$$Lambda$1/20132171@cc34f4d rejected from java.util.concurrent.ThreadPoolExecutor@17a7cec2[Running, pool size = 7, active threads = 7, queued tasks = 0, completed tasks = 0]
当前队列大小:0
at java.util.concurrent.ThreadPoolExecutor$AbortPolicy.rejectedExecution(ThreadPoolExecutor.java:2063)
线程名称:pool-1-thread-1
at java.util.concurrent.ThreadPoolExecutor.reject(ThreadPoolExecutor.java:830)
当前线程数量:2
at java.util.concurrent.ThreadPoolExecutor.execute(ThreadPoolExecutor.java:1379)
当前队列大小:0
at org.tonny.threads.threadpool.ThreadPool.main(ThreadPool.java:19)
线程名称:pool-1-thread-2
当前线程数量:3
当前队列大小:0
线程名称:pool-1-thread-3
当前线程数量:4
当前队列大小:0
当前线程数量:5
当前队列大小:0
线程名称:pool-1-thread-4
当前线程数量:6
当前队列大小:0
当前线程数量:7
当前队列大小:0
线程名称:pool-1-thread-5
线程名称:pool-1-thread-7
线程名称:pool-1-thread-6 Process finished with exit code -1
拥有公平(FIFO)和非公平(LIFO)策略,非公平侧罗会导致一些数据永远无法被消费的情况。使用SynchronousQueue阻塞队列一般要求maximumPoolSizes为无界,避免线程拒绝执行操作。SynchronousQueue没有容量,是无缓冲等待队列,是一个不存储元素的阻塞队列,会直接将任务交给消费者,必须等队列中的添加元素被消费后才能继续添加新的元素。
售票厅- LinkedBlockingQueue
public static void main(String[] args) {
// 核心线程数
int corePoolSize = 3;
//最大线程数
int maximumPoolSize = 7;
// 非核心线程的最大空闲时间
long keepAliveTime = 10L;
// 队列种类
//SynchronousQueue synchronousQueue = new SynchronousQueue<Runnable>();
LinkedBlockingQueue linkedBlockingQueue = new LinkedBlockingQueue<Runnable>();
//ArrayBlockingQueue arrayBlockingQueue = new ArrayBlockingQueue<Runnable>(5);
ThreadPoolExecutor threadpool = new ThreadPoolExecutor(corePoolSize, maximumPoolSize, keepAliveTime, TimeUnit.SECONDS, linkedBlockingQueue);
for (int i = 0; i < 20; i++) {
threadpool.execute(() -> {
System.out.println("线程名称:" + Thread.currentThread().getName());
try {
Thread.sleep(50000L);
} catch (Exception ex) {
ex.printStackTrace();
}
});
System.out.println("当前线程数量:" + threadpool.getPoolSize());
System.out.println("当前队列大小:" + linkedBlockingQueue.size());
}
}
输出结果
当前线程数量:1
当前队列大小:0
当前线程数量:2
当前队列大小:0
线程名称:pool-1-thread-1
线程名称:pool-1-thread-2
当前线程数量:3
当前队列大小:0
当前线程数量:3
当前队列大小:1
当前线程数量:3
当前队列大小:2
当前线程数量:3
当前队列大小:3
当前线程数量:3
当前队列大小:4
当前线程数量:3
当前队列大小:5
当前线程数量:3
当前队列大小:6
当前线程数量:3
当前队列大小:7
线程名称:pool-1-thread-3
当前线程数量:3
当前队列大小:8
当前线程数量:3
当前队列大小:9
当前线程数量:3
当前队列大小:10
当前线程数量:3
当前队列大小:11
当前线程数量:3
当前队列大小:12
当前线程数量:3
当前队列大小:13
当前线程数量:3
当前队列大小:14
当前线程数量:3
当前队列大小:15
当前线程数量:3
当前队列大小:16
当前线程数量:3
当前队列大小:17 Process finished with exit code -1
LinkedBlockingQueue是一个无界缓存等待队列。当前执行的线程数量达到corePoolSize的数量时,剩余的任务会在阻塞队列里等待。(所以在使用此阻塞队列时maximumPoolSizes就相当于无效了),每个线程完全独立于其他线程。生产者和消费者使用独立的锁来控制数据的同步,即在高并发的情况下可以并行操作队列中的数
售票厅- ArrayBlockingQueue
public static void main(String[] args) {
// 核心线程数
int corePoolSize = 3;
//最大线程数
int maximumPoolSize = 7;
// 非核心线程的最大空闲时间
long keepAliveTime = 10L;
// 队列种类
//SynchronousQueue synchronousQueue = new SynchronousQueue<Runnable>();
//LinkedBlockingQueue linkedBlockingQueue = new LinkedBlockingQueue<Runnable>();
ArrayBlockingQueue arrayBlockingQueue = new ArrayBlockingQueue<Runnable>(5);
ThreadPoolExecutor threadpool = new ThreadPoolExecutor(corePoolSize, maximumPoolSize, keepAliveTime, TimeUnit.SECONDS, arrayBlockingQueue);
for (int i = 0; i < 20; i++) {
threadpool.execute(() -> {
System.out.println("线程名称:" + Thread.currentThread().getName());
try {
Thread.sleep(50000L);
} catch (Exception ex) {
ex.printStackTrace();
}
});
System.out.println("当前线程数量:" + threadpool.getPoolSize());
System.out.println("当前队列大小:" + arrayBlockingQueue.size());
}
}
输出结果
当前线程数量:1
当前队列大小:0
线程名称:pool-1-thread-1
当前线程数量:2
当前队列大小:0
当前线程数量:3
当前队列大小:0
当前线程数量:3
当前队列大小:1
当前线程数量:3
当前队列大小:2
当前线程数量:3
当前队列大小:3
线程名称:pool-1-thread-2
当前线程数量:3
当前队列大小:4
线程名称:pool-1-thread-3
当前线程数量:3
当前队列大小:5
当前线程数量:4
当前队列大小:5
当前线程数量:5
当前队列大小:5
线程名称:pool-1-thread-4
当前线程数量:6
当前队列大小:5
线程名称:pool-1-thread-5
当前线程数量:7
当前队列大小:5
线程名称:pool-1-thread-7
线程名称:pool-1-thread-6
Exception in thread "main" java.util.concurrent.RejectedExecutionException: Task org.tonny.threads.threadpool.ThreadPool$$Lambda$1/500977346@4769b07b rejected from java.util.concurrent.ThreadPoolExecutor@cc34f4d[Running, pool size = 7, active threads = 7, queued tasks = 5, completed tasks = 0]
at java.util.concurrent.ThreadPoolExecutor$AbortPolicy.rejectedExecution(ThreadPoolExecutor.java:2063)
at java.util.concurrent.ThreadPoolExecutor.reject(ThreadPoolExecutor.java:830)
at java.util.concurrent.ThreadPoolExecutor.execute(ThreadPoolExecutor.java:1379)
at org.tonny.threads.threadpool.ThreadPool.main(ThreadPool.java:19) Process finished with exit code -1
ArrayBlockingQueue是一个有界缓存等待队列,可以指定缓存队列的大小,当正在执行的线程数等于corePoolSize时,多余的任务缓存在ArrayBlockingQueue队列中等待有空闲的线程时继续执行,当ArrayBlockingQueue已满时,会开启新的线程去执行,当线程数已经达到最大的maximumPoolSizes时,再有新的任务尝试加入ArrayBlockingQueue时会报错。
线程池ThreadPoolExecutor参数分析的更多相关文章
- 线程池ThreadPoolExecutor参数设置
线程池ThreadPoolExecutor参数设置 JDK1.5中引入了强大的concurrent包,其中最常用的莫过了线程池的实现ThreadPoolExecutor,它给我们带来了极大的方便,但同 ...
- 多线程学习笔记八之线程池ThreadPoolExecutor实现分析
目录 简介 继承结构 实现分析 ThreadPoolExecutor类属性 线程池状态 构造方法 execute(Runnable command) addWorker(Runnable firstT ...
- Java 线程池(ThreadPoolExecutor)原理分析与使用
在我们的开发中"池"的概念并不罕见,有数据库连接池.线程池.对象池.常量池等等.下面我们主要针对线程池来一步一步揭开线程池的面纱. 使用线程池的好处 1.降低资源消耗 可以重复利用 ...
- Java线程池(ThreadPoolExecutor)原理分析与使用
在我们的开发中"池"的概念并不罕见,有数据库连接池.线程池.对象池.常量池等等.下面我们主要针对线程池来一步一步揭开线程池的面纱. 使用线程池的好处 1.降低资源消耗 可以重复利用 ...
- Java 线程池(ThreadPoolExecutor)原理分析与实际运用
在我们的开发中"池"的概念并不罕见,有数据库连接池.线程池.对象池.常量池等等.下面我们主要针对线程池来一步一步揭开线程池的面纱. 有关java线程技术文章还可以推荐阅读:< ...
- Java入门系列之线程池ThreadPoolExecutor原理分析思考(十五)
前言 关于线程池原理分析请参看<http://objcoding.com/2019/04/25/threadpool-running/>,建议对原理不太了解的童鞋先看下此文然后再来看本文, ...
- Java线程池ThreadPoolExecutor使用和分析(三) - 终止线程池原理
相关文章目录: Java线程池ThreadPoolExecutor使用和分析(一) Java线程池ThreadPoolExecutor使用和分析(二) - execute()原理 Java线程池Thr ...
- Java线程池ThreadPoolExecutor使用和分析(二) - execute()原理
相关文章目录: Java线程池ThreadPoolExecutor使用和分析(一) Java线程池ThreadPoolExecutor使用和分析(二) - execute()原理 Java线程池Thr ...
- Java线程池ThreadPoolExecutor使用和分析(一)
相关文章目录: Java线程池ThreadPoolExecutor使用和分析(一) Java线程池ThreadPoolExecutor使用和分析(二) - execute()原理 Java线程池Thr ...
随机推荐
- MYSQL进阶学习笔记十二:MySQL 表分区!(视频序号:进阶_29,30)
知识点十三:MySQL 表的分区(29) 一.什么要采用分区: 分区的定义: 当数据量过大的时候(通常是指百万级或千万级数据的时候),这时候需要将一张表划分几张表存储.一些查询可以得到极大的优化,这主 ...
- MYSQL进阶学习笔记七:MySQL触发器的创建,应用及管理!(视频序号:进阶_16,17)
知识点八:MySQL触发器的应用(16,17) 触发器的定义: 什么是触发器: 触发器是一种特殊的存储过程,它在插入,删除或修改特定表中的数据是触发执行,他比数据库本身标准的功能有更精细和更复杂的数据 ...
- classname.this 和 this的使用场景
今天在写代码时,发现在写了一个内部类,而在内部类中需要调用外部类的实例方法,直接使用this调用发现调用的不是外部类而是内部类,于是查找资料原来需要使用外部类的classname.this这样的调用, ...
- BZOJ_2253_[2010 Beijing wc]纸箱堆叠 _CDQ分治+树状数组
BZOJ_2253_[2010 Beijing wc]纸箱堆叠 _CDQ分治+树状数组 Description P 工厂是一个生产纸箱的工厂.纸箱生产线在人工输入三个参数 n p a , , 之后, ...
- Keepalived + HAProxy 搭建【第二篇】Keepalived 的安装与配置
第一步:准备 1. 简介 本文搭建的是利用 Keepalived 实现 HAProxy 的热备方案,即两台主机上的 HAProxy 实例同时运行,其中全总较高的实例为 MASTER,MASTER出现异 ...
- socket入门教程
Server.cs 服务端程序 using System; using System.Collections.Generic; using System.ComponentModel; using ...
- HDU 5901 Count primes (模板题)
题意:给求 1 - n 区间内的素数个数,n <= 1e11. 析:模板题. 代码如下: #pragma comment(linker, "/STACK:1024000000,1024 ...
- h.264的POC计算(转载)
转自:http://www.cnblogs.com/TaigaCon/p/3551001.html 本文参考自http://wenku.baidu.com/link?url=ZPF0iSKzwLQg_ ...
- 洛谷 - P2805 - 植物大战僵尸 - 最大流 - 最大权闭合子图
https://www.luogu.org/problemnew/show/P2805 最大权闭合子图的特点是,假如你要选一个结点,则要先选中它的所有子节点.正权连S负权连T,容量为绝对值,原图有向边 ...
- SCUT - 290 - PARCO的因数游戏 - 博弈论
https://scut.online/p/290 一个 N 个数的取数游戏,Kaildls 和 Parco 轮流操作,每次操作从 N 个数中取一个数 y 并把他变成 y-x(满足 x | y 且x ...