你也被Spring的这个“线程池”坑过吗?

本文讲述了作者在使用Spring的SimpleAsyncTaskExecutor时遇到的一秒多延迟问题,分析了线程池的工作原理,包括创建线程、入任务队列和拒绝策略的影响。通过源码分析发现,SimpleAsyncTaskExecutor并非真正的线程池,而是一个每次执行任务前都会检查并发线程数量的工具,这导致了延迟。作者提醒开发者在使用接口前要了解其实现,遵循代码规约,明确接口职责,避免误解和问题的发生。
摘要由CSDN通过智能技术生成

点击↑上方↑蓝色“编了个程”关注我~

每周至少一篇原创文章

这是本公众号的第 22 篇原创文章

前两天一个晚上,正当我沉浸在敲代码的快乐中时,听到隔壁的同事传来一声不可置信的惊呼:线程池提交命令怎么可能会执行一秒多?

线程池提交方法执行一秒多?那不对啊,线程池提交应该是一个很快的操作,一般情况下不应该执行一秒多那么长的时间。

看了一下那段代码,好像也没什么问题,就是一个简单的提交任务的代码。

executor.execute( () -> {
    // 具体的任务代码
    // 这里有个for循环
});

虽然执行的Job里面有一个for循环,可能比较耗时,但是execute提交任务的时候,并不会去真正去执行Job,所以应该不是这个原因引起的。

分析

看到这个情况,我们首先想到的是线程池提交任务时候的一个处理过程:

线程池原理图

然后逐个分析一下有可能耗时一秒多的操作:

创建线程耗时?

根据上面的图,我们可以知道,如果核心线程数量设置过大,就可能会不断创建新的核心线程去执行任务。同理,如果核心线程池和任务队列都满了,会创建非核心线程去执行任务。

创建线程是比较耗时的,而且Java线程池在这里创建线程的时候还上了锁。

final ReentrantLock mainLock = this.mainLock;
mainLock.lock();

我们写个简单的程序,可以模拟出来线程池耗时的操作,下面这段代码创建2w个线程,在我的电脑里大概会耗时6k多毫秒。

long before = System.currentTimeMillis();
for (int i = 0; i < 20000; i++) {
    // doSomething里面睡眠一秒
    new Thread(() 
  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值