JDK自带线程池学习
JDK自带线程池
线程池的状态
线程有如下状态
- RUNNING状态:Accept new tasks and process queued tasks
- SHUTDOWN状态:Don't accept new tasks, but process queued tasks
- STOP状态: Don't accept new tasks, don't process queued tasks, and interrupt in-progress tasks
- TIDYING状态:All tasks have terminated, workerCount is zero, the thread transitioning to state TIDYING will run the terminated() hook method
- TERMINATED状态:terminated() has completed The numerical order among these values matters, to allow ordered comparisons.
private final AtomicInteger ctl = new AtomicInteger(ctlOf(RUNNING, 0));
private static final int COUNT_BITS = Integer.SIZE - 3;
private static final int CAPACITY = (1 << COUNT_BITS) - 1;
// runState is stored in the high-order bits
private static final int RUNNING = -1 << COUNT_BITS; // -1为全1,所以我们左移29位就是111开头的状态位
private static final int SHUTDOWN = 0 << COUNT_BITS; // 000
private static final int STOP = 1 << COUNT_BITS; // 001
private static final int TIDYING = 2 << COUNT_BITS; // 010
private static final int TERMINATED = 3 << COUNT_BITS; // 011
// Packing and unpacking ctl
private static int runStateOf(int c) { return c & ~CAPACITY; }
private static int workerCountOf(int c) { return c & CAPACITY; }
private static int ctlOf(int rs, int wc) { return rs | wc; }
我们从上面的源代码中可以看出,我们将状态存储在一个原子整型的前三位,然后将线程的容量存储在后29位。将状态和容量放在一起,这样更新状态和容量只需要进行一次cas操作。
下面的三个方法就是进行获取状态和工作线程数量和初始化状态。
在源代码注释中解释到,当未来原子整型不够用了,就会将其升级为原子长整形。且状态位也有扩展的空间,如果需要的话。
同时源代码中也有表明各个状态转换的条件,可以ThreadPoolExecutor类中下载source查看。
线程池构造方法
参数组成
- corePoolSize核心线程的数量
- maximumPoolSize最大的线程数量 PS: 最大线程数-核心线程数 = 急救线程的数量
- keepAliveTime 急救线程的存活时间
- unit 急救线程存活时间单位
- workQueue 阻塞队列
- threadFactory 线程工厂 PS:线程工厂就是创造线程的工厂,为其进行给任务和名字
- handler 拒绝策略的实现 PS:就是当阻塞队列满了之后所要做的动作,死等,限时等(RocketMQ),交给调用者运行,直接抛弃,创建一个新线程(netty),抛出异常写日志(dubbo)。
public ThreadPoolExecutor(int corePoolSize,
int maximumPoolSize,
long keepAliveTime,
TimeUnit unit,
BlockingQueue<Runnable> workQueue,
ThreadFactory threadFactory,
RejectedExecutionHandler handler)
这就是拒绝策略在JDK自带的实现。
- AbortPolicy直接抛出一个RejectedExecutionException异常,dubbo应该是加以记录一些更多的信息 猜测
- CallerRunsPolicy就是让调用者自己执行这个任务
- DiscardPolicy直接抛弃
- DiscardOldestPolicy抛弃早进入阻塞队列的然后让当前任务进入阻塞队列

JDK线程池和上一次的线程池不一样的就是急救线程。
急救线程就是当阻塞队列满了之后,并不会像我上次的例子一样直接进行拒绝策略的判断,会创建一个急救线程或者已存活的急救线程进行执行任务,如果急救线程也满了的话,才会进入拒绝策略的判断。
JDK线程池的基本使用
固定大小线程池
ExecutorService executorService = Executors.newFixedThreadPool(2, new ThreadFactory() {
AtomicInteger ctl = new AtomicInteger();
@Override
public Thread newThread(Runnable r) {
Thread thread = new Thread(r,"myThreadPoll-" + ctl.getAndIncrement());
return thread;
}
});
executorService.execute(()->{
log.debug("线程执行了一次");
});
executorService.execute(()->{
log.debug("线程执行了一次");
});
executorService.execute(()->{
log.debug("线程执行了一次");
});
我看了一下它的默认构造方法,它创造了一个Integer.MAX_VALUE大小的阻塞队列。阻塞队列其实和小测验的阻塞队列是差不多的。
缓存线程池
主要是它的阻塞队列的不同,其中核心数为0,然后通过阻塞队列直到有线程对其进行取任务,不然就是一直阻塞的状态。
@Slf4j
public class Test2 {
public static void main(String[] args) {
ExecutorService executorService = Executors.newCachedThreadPool();
executorService.execute(()->{
log.debug("执行成功");
});
executorService.execute(()->{
log.debug("执行成功");
});
}
}
A blocking queue in which each insert operation must wait for a corresponding remove operation by another thread, and vice versa.
翻译:一种阻塞队列,其中每个插入操作必须等待另一个线程执行相应的删除操作,反之亦然。
单线程线程池
顾名思义即单线程的线程池,不过有意思的一点就是这个使用了一个设计模式就是装饰器模式。
@Slf4j
public class Test3 {
public static void main(String[] args) {
ExecutorService executorService = Executors.newSingleThreadExecutor();
executorService.execute(()->{
log.debug("hello");
});
executorService.execute(()->{
log.debug("hello");
});
}
}
首先是因为如果我们直接返回ThreadPoolExecutor这个类的话,我们是知道了它的类,是可以直接使用强转来实现修改线程的核心数以及一些参数。如下
@Slf4j
public class Test1 {
public static void main(String[] args) {
ExecutorService executorService = Executors.newFixedThreadPool(2, new ThreadFactory() {
AtomicInteger ctl = new AtomicInteger();
@Override
public Thread newThread(Runnable r) {
Thread thread = new Thread(r,"myThreadPoll-" + ctl.getAndIncrement());
return thread;
}
});
// 我们将其对象通过强转直接修改了其的核心数,执行结构同样生效
ThreadPoolExecutor executor= (ThreadPoolExecutor) executorService;
executor.setCorePoolSize(1);
executorService.execute(()->{
log.debug("线程执行了一次");
});
executorService.execute(()->{
log.debug("线程执行了一次");
});
executorService.execute(()->{
log.debug("线程执行了一次");
});
}
}
但是如果我们通过装饰器模式将其进行包装,然后包装的对象返回,是无法进行修改核心数的,更何况单线程线程池的情况下,我们需要保证核心数总为1把,不能让其他人修改。展示部分代码,返回的是其的装饰类。
static class DelegatedExecutorService extends AbstractExecutorService {
private final ExecutorService e;
DelegatedExecutorService(ExecutorService executor) { e = executor; }
public void execute(Runnable command) { e.execute(command); }
public void shutdown() { e.shutdown(); }
public List<Runnable> shutdownNow() { return e.shutdownNow(); }
ExecutorService方法使用
这个类就是线程池的接口类,掌管着线程池的方法。
- void shutdown() 继续执行当前线程池中的任务和阻塞队列中的任务,不再接收新的任务。
- List shutdownNow() 尝试停止所有正在执行的任务,停止正在等待的任务的处理,并返回正在等待执行的任务的列表
- boolean isShutdown() 返回当前线程池是否已经关闭
- boolean isTerminated() 在调用了shutdown或Shutdownow后,关闭后所有任务都已完成,则返回true。如果没有调用永远不会返回true。
- boolean awaitTermination(long timeout, TimeUnit unit) 阻塞,直到所有任务在关闭请求后完成执行,或超时发生,或当前线程中断,以先发生的为准。
- Future submit(Callable task) 返回执行结果
- List<Future> invokeAll(Collection<? extends Callable> tasks) 批量返回结果
- T invokeAny(Collection<? extends Callable> tasks) 执行任意一个并返回结果,如果任何一个完成,其他正在执行的直接结束。
工作线程的饥饿现象
/**
* @Author 10276
* @Date 2022/5/11 20:52
*/
@Slf4j
public class Test4 {
public static void main(String[] args) {
ExecutorService rest = Executors.newFixedThreadPool(2);
rest.execute(()->{
log.debug("准备点餐");
Future<String> submit = rest.submit(() -> {
log.debug("正在坐菜");
return "菜";
});
try {
log.debug("上菜{}",submit.get());
} catch (InterruptedException | ExecutionException e) {
e.printStackTrace();
}
});
rest.execute(()->{
log.debug("准备点餐");
Future<String> submit = rest.submit(() -> {
log.debug("正在坐菜");
return "菜";
});
try {
log.debug("上菜{}",submit.get());
} catch (InterruptedException | ExecutionException e) {
e.printStackTrace();
}
});
}
}
一个线程池的导致两个线程没办法继续进行下去,没有多余的线程取做接下来的事情,但这不是死锁问题,归结原因还是线程资源不够,同时也无法继续进行下去了。

解决方案:由此可以得出对于线程池数量的选择和线程池中核心线程数量的选择是十分重要的。
public static void main(String[] args) {
ExecutorService waitress = Executors.newFixedThreadPool(1);
ExecutorService cooker = Executors.newFixedThreadPool(1);
waitress.execute(()->{
log.debug("准备点餐");
Future<String> submit = cooker.submit(() -> {
log.debug("正在做菜");
return "湖南菜";
});
try {
log.debug("上菜:{}",submit.get());
} catch (InterruptedException | ExecutionException e) {
e.printStackTrace();
}
});
waitress.execute(()->{
log.debug("准备点餐");
Future<String> submit = cooker.submit(() -> {
log.debug("正在做菜");
return "广东菜";
});
try {
log.debug("上菜:{}",submit.get());
} catch (InterruptedException | ExecutionException e) {
e.printStackTrace();
}
});
}
JDK自带线程池学习的更多相关文章
- JDK自带线程池介绍及使用环境
1.newFixedThreadPool创建一个指定工作线程数量的线程池.每当提交一个任务就创建一个工作线程,如果工作线程数量达到线程池初始的最大数,则将提交的任务存入到池队列中. 2.newCach ...
- Java线程池学习
Java线程池学习 Executor框架简介 在Java 5之后,并发编程引入了一堆新的启动.调度和管理线程的API.Executor框架便是Java 5中引入的,其内部使用了线程池机制,它在java ...
- 【Java多线程】线程池学习
Java线程池学习 众所周知,Java不仅提供了线程,也提供了线程池库给我们使用,那么今天来学学线程池的具体使用以及线程池基本实现原理分析. ThreadPoolExecutor ThreadPool ...
- Java高并发程序设计学习笔记(六):JDK并发包(线程池的基本使用、ForkJoin)
转自:https://blog.csdn.net/dataiyangu/article/details/86573222 1. 线程池的基本使用1.1. 为什么需要线程池1.2. JDK为我们提供了哪 ...
- 从源码看JDK提供的线程池(ThreadPoolExecutor)
一丶什么是线程池 (1)博主在听到线程池三个字的时候第一个想法就是数据库连接池,回忆一下,我们在学JavaWeb的时候怎么理解数据库连接池的,数据库创建连接和关闭连接是一个比较耗费资源的事情,对于那些 ...
- (转)java自带线程池和队列详细讲解 - CSDN过天的专栏
一简介 线程的使用在java中占有极其重要的地位,在jdk1.4极其之前的jdk版本中,关于线程池的使用是极其简陋的.在jdk1.5之后这一情况有了很大的改观.Jdk1.5之后加入了java.util ...
- java自带线程池和队列详细讲解
Java线程池使用说明 一简介 线程的使用在java中占有极其重要的地位,在jdk1.4极其之前的jdk版本中,关于线程池的使用是极其简陋的.在jdk1.5之后这一情况有了很大的改观.Jdk1.5之后 ...
- JAVA线程池学习,ThreadPoolTaskExecutor和ThreadPoolExecutor有何区别?
初学者很容易看错,如果没有看到spring或者JUC源码的人肯定是不太了解的. ThreadPoolTaskExecutor是spring core包中的,而ThreadPoolExecutor是JD ...
- c++11 线程池学习笔记 (一) 任务队列
学习内容来自一下地址 http://www.cnblogs.com/qicosmos/p/4772486.html github https://github.com/qicosmos/cosmos ...
随机推荐
- 使用缓存(Cache)的几种方式,回顾一下~~~
前言 如今缓存成为了优化网站性能的首要利器,缓存使用的好,不仅能让网站性能提升,让用户体验变好,而且还能节约成本(增加一台缓存服务器可能就节约好几台机器):那平时小伙伴们都使用哪些缓存方式呢?这里就来 ...
- 业务网关之AK中心建设
啥是AK AK(Access Key)是一种身份证明,它解决了"资源的使用者是谁"这个问题,比如在生活中,身份证可以证明你是你,而在云计算或程序中,AK能证明你是这个应用的拥有者. ...
- 理解OIDC协议和认证机制
当互联网应用越来越多,每个应用程序都实现了自己的身份存储.认证和授权,用户需要在应用上反复的注册与登录,体验糟糕,用户身份信息无法在多个应用间共享与同步.当使用企业应用时,企业提供了一系列应用,尽管是 ...
- Numpy实现机器学习交叉验证的数据划分
Numpy实现K折交叉验证的数据划分 本实例使用Numpy的数组切片语法,实现了K折交叉验证的数据划分 背景:K折交叉验证 为什么需要这个?在机器学习中,因为如下原因,使用K折交叉验证能更好评估模型效 ...
- 模型预测控制(MPC)简介
1.引言 在当今过程控制中,PID当然是用的最多的控制方法,但MPC也超过了10%的占有率.MPC是一个总称,有着各种各样的算法.其动态矩阵控制(DMC)是代表作.DMC采用的是系统的阶跃响应曲线,其 ...
- 14_Nonlinear Basic Feedback Stabilization_非线性系统稳定性设计
非线性系统线性化的方式:泰勒展开近似线性化(2_线性化_泰勒级数_泰勒公式_Linearization).反馈线性化,本文使用的是反馈线性化 从图中可知道输入u非常大达到了900多,所以直接使用u消去 ...
- PCB产业链、材料、工艺流程详解(1)
PCB知识大全 1.什么是pcb,用来干什么? PCB( Printed Circuit Board),中文名称为印制电路板,又称印刷线路板,是重要的电子部件,是电子元器件的支撑体,是电子元器件电气连 ...
- 虚拟机VMware的安装与Xshell的应用
先安装VMware 1.安装就按照提示一点点安装就行了 配置网络 打开VMware 这里的IOS映像文件在https://developer.aliyun.com/mirror/里下载 这里用方向键往 ...
- QT类使用记录
QT类使用记录 1.QSharedMemory 提供了对一段共享内存的访问.既提供了被多进程和多线程共享的一段内存的访问.也为单线程或单进程锁定内存以实现互斥访问提供了方法. QSharedMemor ...
- Go xmas2020 学习笔记 10、Slices in Detail
10-Slices in Detail. Slice. Empty vs nil slice