线程池:ThreadPoolExcutor源码阅读
ThreadPoolExcutor源码流程图:(图片较大,下载再看比较方便)

线程池里的二进制奥秘
前言:
线程池的五种状态state(RUNNING、SHUTDOWN、STOP、TIDYING、TERMINATED)和线程池的工作线程数:workerCount,
这两个变量,可有通过一个变量ctl转成二进制后而获得。
直接看线程池ThreadPoolExecutor源码里,管理状态和工作线程数的代码
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; private static final int RUNNING = -1 << COUNT_BITS;
private static final int SHUTDOWN = 0 << COUNT_BITS;
private static final int STOP = 1 << COUNT_BITS;
private static final int TIDYING = 2 << COUNT_BITS;
private static final int TERMINATED = 3 << COUNT_BITS; 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; }
Integer.size为Integer类型最大值的二进制的位数:32(位),COUNT_BITS = 29(位),为啥是要要减去3,拿个29呢,因为它想用二进制高3位来表示线程池的状态(state),后面的29位用来表示工作线程数量(workerCount)。
比如RUNNINGZ状态是 -1左移29位,即:
-1的二进制算法,先取绝对值1的二进制:00000000 00000000 00000001,再取反码:11111111 11111111 11111110
再取补码(加1):11111111 11111111 11111111 ,这就是-1的二进制表示。
左移运算:-1<<29 = 11100000 00000000 00000000 (左移29位, 移动多少位, 后面补多少个0, 高于32位的去掉)。
各个state:
-1<<29 = 11100000 00000000 00000000 => RUNNINGZ
0<< 29 = 00000000 00000000 00000000 => SHUTDOWN
1 <<29 = 00100000 00000000 00000000 => STOP
2 <<29 = 01000000 00000000 00000000 => TIDYING
3 <<29 = 01100000 00000000 00000000 => TERMINATED
线程池就是这样用高三位来表示不同的状态。
所以状态还有这样的关系: RUNNINGZ < SHUTDOWN < STOP < TIDYING < TERMINATED
不管各个状态低29位是啥,这个关系都不会变。
再看CAPACITY = (1<<29) -1:
即00100000 00000000 00000000 - 1 = 00011111 11111111 11111111 ,这是线程池最大容量线程数(536870911),
即是用1到29位可表示线程的数量,那么,ctl = state.高3位 + 低29位,如:
ctl = 11111111 11111111 11111111,表示线程池是RUNNINGZ状态,且线程数为536870911;
ctl = 00000000 00000000 00000001,表示线程池是SHUTDOWN状态,且线程数为1;
CAPACITY=536870911,表示线程池最大线程数容量为536870911,但这个最大数值可以忽略,
因为在实例化线程池的时候,已经通过传参数设置了核心线程数、允许最大线程数。
初始化为 ctl = new AtomicInteger(ctlOf(RUNNING, 0));
ctlOf(RUNNING, 0)参数中的0,意思是初始化线程池的线程数为0。然后通过ctlOf做或运算,即:
11100000 00000000 00000000 | 00000000 00000000 00000000 = 11100000 00000000 00000000
运算后的结果,把表示状态的高3位、表示线程数的低29位组合起来得到一个数ctl,
下次拿到ctl之后,就知道此时线程池的状态和线程数量了。
1、那么runStateOf(int c)方法具体是怎么知道线程池的状态的?
假设c = ctl.get() = 00100000 00000000 00000011,表示STOP状态且还有3个线程
CAPACITY = (1<<29)- 1 = 00100000 00000000 00000000 - 1 = 00011111 11111111 11111111
为啥要减1呢?因为减1后:
1、把表示状态的高3位给挪留出来了,值又可以表示线程池最大线程数。
2、把表示线程数的低29位变为了1,那 ~CAPACITY = 11100000 00000000 00000000, 所以:
c &~CAPACITY =00100000 00000000 00000011 & 11100000 00000000 00000000 = 00100000 00000000 00000000,
结果00100000 00000000 00000000保持c的高3位不变,把c的低29位全变成0,这就是获得线程池是STOP状态
2、通过workerCountOf(int c)获取工作线程数,同上理
线程池:ThreadPoolExcutor源码阅读的更多相关文章
- 深入浅出Java线程池:源码篇
前言 在上一篇文章深入浅出Java线程池:理论篇中,已经介绍了什么是线程池以及基本的使用.(本来写作的思路是使用篇,但经网友建议后,感觉改为理论篇会更加合适).本文则深入线程池的源码,主要是介绍Thr ...
- Java并发包源码学习系列:线程池ScheduledThreadPoolExecutor源码解析
目录 ScheduledThreadPoolExecutor概述 类图结构 ScheduledExecutorService ScheduledFutureTask FutureTask schedu ...
- java线程池ThreadPoolExector源码分析
java线程池ThreadPoolExector源码分析 今天研究了下ThreadPoolExector源码,大致上总结了以下几点跟大家分享下: 一.ThreadPoolExector几个主要变量 先 ...
- linux线程池thrmgr源码解析
linux线程池thrmgr源码解析 1 thrmgr线程池的作用 thrmgr线程池的作用是提高程序的并发处理能力,在多CPU的服务器上运行程序,可以并发执行多个任务. 2 ...
- Java并发包源码学习系列:线程池ThreadPoolExecutor源码解析
目录 ThreadPoolExecutor概述 线程池解决的优点 线程池处理流程 创建线程池 重要常量及字段 线程池的五种状态及转换 ThreadPoolExecutor构造参数及参数意义 Work类 ...
- 一个python线程池的源码解析
python为了方便人们编程高度封装了很多东西,比如进程里的进程池,大大方便了人们编程的效率,但是默认却没有线程池,本人前段时间整理出一个线程池,并进行了简单的解析和注释,本人水平有限,如有错误希望高 ...
- Java调度线程池ScheduledThreadPoolExecutor源码分析
最近新接手的项目里大量使用了ScheduledThreadPoolExecutor类去执行一些定时任务,之前一直没有机会研究这个类的源码,这次趁着机会好好研读一下. 该类主要还是基于ThreadPoo ...
- [转载] Java线程池框架源码分析
转载自http://www.linuxidc.com/Linux/2014-11/108791.htm 相关类Executor,Executors,AbstractExecutorService,Ex ...
- 线程池ThreadPoolExecutor源码解读研究(JDK1.8)
一.什么是线程池 为什么要使用线程池?在多线程并发开发中,线程的数量较多,且每个线程执行一定的时间后就结束了,下一个线程任务到来还需要重新创建线程,这样线程数量特别庞大的时候,频繁的创建线程和销毁线程 ...
随机推荐
- python循环与基本数据类型内置方法
今天又是充满希望的一天呢 一.python循环 1.wuile与else连用 当while没有被关键'break'主动结束的情况下 正常结束循环体代码之后会执行else的子代码 "" ...
- Sql递归查询,Sqlserver、Oracle、PG、Mysql
递归分两种:一种由父项向下级递归,另一种是由子项向上级递归.下面就这两种情况做个简单的处理. 假设有一个表treeview,包含字段 id,parentid,text 分别代表id,上级id,描述字段 ...
- VUE(uni-app)+SSM 微信小程序
环境 jdk:1.8.0_181 tomcat:7.0.100 mysql:8.0.25 工具 ideaIU+Hbuilder 遇到的问题 1.需要跳转到注册在tobar中页面时,需使用 uni.sw ...
- 2021-ACM-ICPC-济南站 K Search For Mafuyu 【树的遍历与回溯,dfs, 邻接表】
PAT 题目详情 (pintia.cn) 题目 题意描述 n个房间, 有n-1个通道, kanade初始在1,Mafuyu 位置未知,求K到M的最小期望,也就是到每个地方的平均值,注意的是这里的求法, ...
- Rust如何开发eBPF应用(一)?
前言 eBPF是一项革命性的技术,可以在Linux内核中运行沙盒程序,而无需重新编译内核或加载内核模块.它能够在许多内核 hook 点安全地执行字节码,主要应用在云原生网络.安全.跟踪监控等方面. e ...
- 被人DDoS攻击了,分析一下原理和防护
一.行业现象 1.1 为什么要攻击? 常见的,一个是同行恶意竞争,一个是敲诈勒索. 无论是传统行业的线下门店,还是互联网行业的门户网站.APP产品,都存在着竞争关系,争相获得更多客源,究其目的,无非是 ...
- Halo 开源项目学习(一):项目启动
项目简介 Halo 是一个优秀的开源博客发布应用,在 GitHub 上广受好评,正好最近在练习写博客,借此记录一下学习 Halo 的过程. 项目下载 从 GitHub 上拉取项目源码,Halo 从 1 ...
- python学习-Day34
目录 今日内容详细 作业讲解 客户端 服务端 UDP协议 实现 服务器端 客户端 基于UDP实现简易版本的qq 客户端 服务器端 操作系统的发展史 第一代计算机(1940~1955) 第二代就算机(1 ...
- java自学中出现的问题或者?
自学java之路,是如此的坎坷.经过一段时间的自学,我得出一些总结! 总结如下: 1. 在学习编程之路(Java)的,最基本的还是学习之路,对编程前程深感迷茫2. 网络中有许许多多的编程 ...
- vmware ubuntu 忘记密码
1.进入recovery模式 修改root密码 1.启动ubuntu系统,一开始在有进度条的时候按下shift键,出现GRUB选择菜单,选择Advanced options for Ubuntu 按回 ...