面试的时候没答对,仔细复盘了一下源码,记录一下线程池执行的流程如下:
1、当一个任务到来的时候会判断核心线程数是否已经满了,(注意核心线程数是否会被回收是根据 参数 allowCoreThreadTimeout()来确定,默认为false,即核心线程数并不会被回收,设置为true时则会与常规线程一样当keepalive time到达的时候会被回收)未满则分配一个核心线程,满了则放置到等待队列中去
2、当队列未满则会将任务添加到队列当中去,满了则会判断是否达到最大线程数
3、达到了最大线程数的数量就会出发rejection策略对任务记性拒绝策略操作,未达到最大线程数量就会创建一个新的常规线程(常规线程会在空闲时被回收)
示例代码:
public static void main(String[] args) {
ThreadPoolExecutor executor = new ThreadPoolExecutor(5,
10,
0,
TimeUnit.MICROSECONDS,
new ArrayBlockingQueue<>(3)
);
for (int i=0;i<10;i++){
executor.submit(new Job()) ;
}
executor.shutdown();
}
执行结果:
5个核心线程执行5个任务,另外队列中等待三个任务,剩下的两个任务创建常规线程去执行
,所以最先出现的是7个data,之后才会出现剩下的三个data,也就是等待队列里的三个任务被取出执行