ThreadPoolExecutor
线程池
为什么要使用线程池?
1,降低资源消耗,因为线程是很宝贵的资源,反复创建会降低系统的性能
2,提升系统的响应速度,通过复用线程,省去创建线程的过程,因此整体上提升了系统的响应速度。
3,提升线程的客观理性,直接创建线程无法统一进行管理,可以使用线程池进行统一调度。
线程池执行任务的过程:
data:image/s3,"s3://crabby-images/50795/507953445c4c2bd365a05203f73158d42320b50d" alt=""
创建线程池的方法或者说线程池的种类很多,但是归根结底,都是调用了下面的这个构造方法。
data:image/s3,"s3://crabby-images/7e8d4/7e8d4df1c428bf2c0e03130494ce418f0c9e9839" alt=""
该方法参数说明如下:
data:image/s3,"s3://crabby-images/d1533/d15331ff8f39de7fcb059e95739a917254eb3e52" alt=""
可以看到该类提供了四种丢弃策略,也可以继承RejectedExecutionHandler去自定义。
data:image/s3,"s3://crabby-images/cd9b0/cd9b0008d7ce105f333a964af3d08130e48ecd03" alt=""
ececute()方法源码分析:
data:image/s3,"s3://crabby-images/7ca36/7ca36f61926c531ff9acb41d004c6972127544c3" alt=""
data:image/s3,"s3://crabby-images/0973a/0973a3528f218298f3fb2d8eea01ef1ec758a705" alt=""
这里的ctl保存了线程池中的线程数以及线程池的状态:线程池的状态共有五种,因此用3位二进制表示。用其他29位表示线程池中的活着的线程数。从表示状态需要左移可以看出,ctl是用高三位表示线程池的状态,用低29位表示线程数。
data:image/s3,"s3://crabby-images/bbef4/bbef4af1349a8f872f53cdc95da020d6ac61613b" alt=""
如何获取线程池中的线程数,就相当于获取低29位代表的数字。
data:image/s3,"s3://crabby-images/77849/778496cbea01fb1d5077afc53e8b60a9aea1d690" alt=""
CAPACITY是对1左移29位,相当于1后面有29个0,再减1相当于有29个1。
0010 0000 0000 0000 0000 0000 0000 0000
-1:0001 1111 1111 1111 1111 1111 1111 1111
在workerCount中,直接用c与CAPACITY就是取c的低29位。在runStateOf中先是对CAPACITY取反,也就是 1110 0000 0000 0000 0000 0000 0000 0000,然后再与c进行与运算,得到的就是c的高三位。
ctlOf是将两个int的进行或运算,包装为ctl。
这个与读写锁中的同步状态记录读锁获取次数和写锁获取次数是一致的。
RejectedHandlerPolicy
AbortPolicy:直接抛出异常
CallerRunsPolicy:A handler for rejected tasks that runs the rejected task directly in the calling thread of the {@code execute} method, unless the executor has been shut down, in which case the task is discarded.
data:image/s3,"s3://crabby-images/3d081/3d0812def7390dedcef2c2937387ed2b45f042ea" alt=""
DiscardPolicy:A handler for rejected tasks that silently discards the rejected task.
具体实现:
data:image/s3,"s3://crabby-images/a3d2d/a3d2d5420bec39189731d8e0695e16e0b2d6da03" alt=""
DiscardOldestPolicy:A handler for rejected tasks that discards the oldest unhandled
* request and then retries {@code execute}, unless the executor
* is shut down, in which case the task is discarded.
data:image/s3,"s3://crabby-images/928e5/928e5bd84c0235d3c57c3679be267e585fa92614" alt=""