一个demo让你彻底搞懂线程池工作流程(看不懂评论区留言喷我)

简介: 线程池无非就那几个参数,啥核心线程、最大线程、队列,没啥难的,有手就能学废来,看完这篇文章还不懂的评论区留言喷我

线程池无非就那几个参数,啥核心线程、最大线程、队列,没啥难的,有手就能学废

来,看完这篇文章还不懂的评论区留言喷我

我直接上demo,不知道参数啥意思的先去隔壁补补课,虽然本文也会提到,你最好先大概知道点,线程池实现运行机制总结

看完机制接着demo

public class ThreadPoolExecutorTest {
    private static final int taskCount = 50;//任务数
    public static void main(String[] args) throws InterruptedException {
        AtomicInteger integer = new AtomicInteger();
        ThreadPoolExecutor executor = new ThreadPoolExecutor(
                10,//核心线程
                20,//最大线程
                5,//非核心回收超时时间
                TimeUnit.SECONDS,//超时时间单位
                new ArrayBlockingQueue<>(30)//任务队列);
        System.out.println("总任务数:" + taskCount);
        long start = System.currentTimeMillis();
        //模拟任务提交
        for (int i = 0; i < taskCount; i++) {
            Thread thread = new Thread(() -> {
                try {
                    Thread.sleep(500);//模拟执行耗时
                    System.out.println("已执行" + integer.addAndGet(1) + "个任务");
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            });
            try {
              //注意这里我try起来了,因为默认拒绝策略会报错
                executor.execute(thread);
            } catch (Exception e) {
                System.out.println(e.getMessage());
            }
        }
        long end = 0;
        while (executor.getCompletedTaskCount() < 50) {
            end = System.currentTimeMillis();
        }
        System.out.println("任务总耗时:" + (end - start));
    }
}

带着问题来看demo


如上,new了个线程池,core线程数10,最大线程数20,任务队列30,请听题!!


问题0:提交5个任务,总耗时多少?


分析:我核心线程数10,也就是说10个线程会长期处于活跃状态,来任务立马能执行,5<10,所以5个任务立马全部执行,多线程并行当然是异步,所以是500ms

ac95551b6a7e4f3cac678aec6aef1d8a.png

插个嘴:为什么不是500而是540,因为代码执行需要花时间,毕竟是模拟提交任务,并不是真正一瞬间提交完

问题1:提交10个任务,总耗时多少?

依然是500,10个任务和5个任务其实都一样,没超过核心线程数,来一个执行一个

7b33f218823740b592384bd1a38a51f0.png

问题2:提交11个任务,总耗时多少?

1000,别惊讶,这就是很多人没搞懂线程池机制的关键点,虽然只多了一个任务,但是第11个任务不会马上执行,因为队列没满,所以前10个任务会立马执行,而第11个会被扔到队列中,等有线程空出来了再执行

e85dd22de38d42778fb4cfb526d3498d.png

问题3:提交20个任务,总耗时多少?

也是1000,别问为什么也别杠,今天就是耶稣来了它也是1000,这20个任务,前10个任务来一个执行一个,从第11个到第20个会全部丢进队列,当前十个任务有任务执行完了,才会从队列取出执行

e85dd22de38d42778fb4cfb526d3498d.png

问题4:提交30个任务,总耗时多少?

当然是1500啦,30/10=3,3*500=1500

54d2b511f0384fe1a4c889bda4d278fa.png

问题5:提交40个任务,总耗时多少?

当然是2000啦,10+10+10+10,我不想解释

484e59dd1395494995f525ac6e56e0d9.png

重点来了!!

重点来了!!

重点来了!!

问题6:提交41个任务,总耗时多少?

也是2000,很多人会认为是1500,11+11+11+8,其实不然,这里先记下,我后面说,先继续往下看

cebaf769f44c486fabb2683be0dfabb8.png


问题7:提交45个任务,总耗时多少?

1500,没错就是1500,15+15+15

95e0e7e46d424b37be2d4df2e72a59d6.png

问题8:提交50个任务,总耗时多少?

1500,20+20+10

c9250986a8414670bba5aa7b50473008.png

问题9:提交51个任务,总耗时多少?

也是1500,这个线程池同时最大接收50个任务,因为我没设置拒绝策略,默认是AbortPolicy,即超出的任务会被丢弃并抛出RejectedExecutionException异常,demo中没报错是因为我try了


最后来说刚才的遗留问题,为啥41个任务2000,45个任务就1500


其实很多人没把这个搞懂的,后面几个问题我都写了个一串加号


如40个任务时,10+10+10+10,这代表所有任务分4组完成,每组执行10个,因为多线程是异步,所以每组执行时间就等于单个任务执行时间,即500ms,所以40个任务就是500+500+500+500=2000


而41个任务时,是11+11+11+8,所以40个任务也是500+500+500+500=2000(肯定会有小伙伴问,41个任务已经超出了队列容量,线程池中线程为啥没达到最大线程数,应该是20+20+1才对啊)


记住一句话

任务数 <= 核心线程数时,线程池中工作线程数 = 任务数

核心线程数 + 队列容量 < 任务数 <= 最大线程数 + 队列容量时,工作线程数 = 任务数 - 队列容量


所以


再来继续看


41个任务时,41-30=11,执行批次为11+11+11+8,即500+500+500+500=2000,有问题吗?没有问题


45个任务时,45-30=15,所以15+15+15,即500+500+500=1500,有问题吗?没有问题


50个任务时,50-30=20,20+20+10,即500+500+500=1500,有问题吗?依然没有问题


课后留个问题,44个任务耗时多少?


希望你自己去跑,别不识抬举


文末再说个冷知识,核心线程数也可以被回收,ThreadPoolExecutor有个属性叫 allowCoreThreadTimeOut


/

ac53d518628547b1b3e37ca36d96ff81.png

ThreadPoolExecutor给我们提供了一个public方法allowCoreThreadTimeOut,通过**allowCoreThreadTimeOut( true )**就能设置

3035be9aa57e42bda6d870b666e7a892.png


博客是今天写的

demo是昨天想的

头发是前天想demo时掉的

相关文章
|
2月前
|
缓存 监控 Java
Java线程池提交任务流程底层源码与源码解析
【11月更文挑战第30天】嘿,各位技术爱好者们,今天咱们来聊聊Java线程池提交任务的底层源码与源码解析。作为一个资深的Java开发者,我相信你一定对线程池并不陌生。线程池作为并发编程中的一大利器,其重要性不言而喻。今天,我将以对话的方式,带你一步步深入线程池的奥秘,从概述到功能点,再到背景和业务点,最后到底层原理和示例,让你对线程池有一个全新的认识。
65 12
|
7月前
|
Java
解析Java线程池:参数详解与执行流程
解析Java线程池:参数详解与执行流程
80 1
|
4月前
|
消息中间件 存储 前端开发
面试官:说说停止线程池的执行流程?
面试官:说说停止线程池的执行流程?
65 2
面试官:说说停止线程池的执行流程?
|
4月前
自己动手写QT多线程demo
本文是作者关于如何编写Qt多线程demo的教程,介绍了如何实现多线程功能,包括可暂停和继续的功能。文章提供了部分示例代码,展示了如何创建线程类、启动和管理线程,以及线程间的通信。同时,还提供了相关参考资料和免费下载链接。
112 0
|
6月前
|
存储 安全 Java
Java面试题:如何在Java应用中实现有效的内存优化?在多线程环境下,如何确保数据的线程安全?如何设计并实现一个基于ExecutorService的任务处理流程?
Java面试题:如何在Java应用中实现有效的内存优化?在多线程环境下,如何确保数据的线程安全?如何设计并实现一个基于ExecutorService的任务处理流程?
54 0
|
8月前
|
Linux API 调度
xenomai内核解析-xenomai实时线程创建流程
本文介绍了linux硬实时操作系统xenomai pthread_creta()接口的底层实现原理,解释了如何在双内核间创建和调度一个xenomai任务。本文是基于源代码的分析,提供了详细的流程和注释,同时给出了结论部分,方便读者快速了解核心内容。
198 0
xenomai内核解析-xenomai实时线程创建流程
|
存储 安全 Java
【线程池添加工作线程的流程】
【线程池添加工作线程的流程】
103 0
|
8月前
|
Python
Python 多线程运用 demo
这是一个Python多线程示例,创建了两个线程`t1`和`t2`分别执行`print_numbers`(打印0-9)和`print_letters`(打印&#39;a&#39;-&#39;j&#39;)函数。通过`start()`启动线程,`join()`确保线程执行完毕后输出&quot;程序结束&quot;。
33 2
|
Java 编译器
Java多线程(4)---死锁和Synchronized加锁流程
Java多线程(4)---死锁和Synchronized加锁流程
81 0
|
8月前
|
存储 NoSQL Redis
深入浅出Redis(二):Redis单线程模型与通信流程
深入浅出Redis(二):Redis单线程模型与通信流程