线程池ThreadPoolExecutor分析: 线程池是什么时候创建线程的,队列中的任务是什么时候取出来的?

Posted yszzu

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了线程池ThreadPoolExecutor分析: 线程池是什么时候创建线程的,队列中的任务是什么时候取出来的?相关的知识,希望对你有一定的参考价值。

 

带着几个问题进入源码分析:

线程池是什么时候创建线程的?

任务runnable task是先放到core到maxThread之间的线程,还是先放到队列?

队列中的任务是什么时候取出来的?

什么时候会触发reject策略?

core到maxThread之间的线程什么时候会die?

task抛出异常,线程池中这个work thread还能运行其他任务吗?

 

至少在new ThreadPoolExecutor()时,Thread对象并没有初始化. 这里仅仅指定了几个初始参数

技术分享图片

一段基础代码,进入分析

public static void main(String[] args) {

        ExecutorService executorService = new ThreadPoolExecutor(2, 5, 0, TimeUnit.DAYS,
                new ArrayBlockingQueue<>(1), new ThreadFactory() {
            @Override
            public Thread newThread(Runnable r) {
                Thread thread = new Thread(r);
//                thread.setDaemon(true);
                return thread;
            }
        });
        // 对象创建后,线程实际还没开始创建

        // 执行execute时,检查当前池中线程数大小是否小于core number, 如果是,则创建新线程
        executorService.execute(() -> {
            System.out.println("[email protected]" + Thread.currentThread().getName());
            sleepTime();
            System.out.println(1);
        });

        //检查当前池中线程数大小是否小于core number, 如果是,则创建新线程
        executorService.execute(() -> {
            System.out.println("[email protected]" + Thread.currentThread().getName());
            sleepTime();
            System.out.println(2);
        });

        // 检查当前池中线程数大小是否小于core number, 如果不是,则偿试放入队列
        // 这个任务是加到队列去的, 注意队列大小只有1,
        // TODO 队列中的任务是什么时候取出来的?
        executorService.execute(() -> {
            System.out.println("[email protected]" + Thread.currentThread().getName());
            sleepTime();
            System.out.println(3);
        });

        // 检查当前池中线程数大小是否小于core number, 如果不是,则偿试放入队列,放入队列也失败,则增加新的worker线程
        // 这个任务是加到core以外的新线程去的
        executorService.execute(() -> {
            System.out.println("[email protected]" + Thread.currentThread().getName());
            sleepTime();
            System.out.println(4);
        });
    }

  

第3行,创建一个核心池2, 最大池5, 队列为1的线程池

执行第一个execute时,进入调试jdk源码

 

技术分享图片代码块1

第一个if, 判断如果当前线程数小于corePoolSize, 则创建新的核心worker对象(Worker中指向Thread对象,保持引用,保证不会被GC回收)

 

我们的示例代码中,第1和第2个线程都是这样创建出线程的

 


以上是关于线程池ThreadPoolExecutor分析: 线程池是什么时候创建线程的,队列中的任务是什么时候取出来的?的主要内容,如果未能解决你的问题,请参考以下文章

线程池源码分析之ThreadPoolExecutor

线程池ThreadPoolExecutor源码分析

线程池ThreadPoolExecutor分析

Java并发编程:Java线程池核心ThreadPoolExecutor的使用和原理分析

Java并发编程21线程池ThreadPoolExecutor源码解析

ThreadPoolExecutor(线程池)源码分析