首页
关于这个博客
Search
1
Java 实现Google 账号单点登录(OAuth 2.0)全流程解析
824 阅读
2
Spring AI 无法获取大模型深度思考内容?解决方案来了
361 阅读
3
EasyExcel 实战:导出带图片的 Excel 完整方案
169 阅读
4
微信小程序实现页面返回前确认弹窗:兼容左上角返回与右滑返回
155 阅读
5
服务器遭遇 XMRig 挖矿程序入侵排查与清理全记录
153 阅读
Java 核心
框架与中间件
数据库技术
开发工具与效率
问题排查与踩坑记录
程序员成长与思考
前端
登录
Search
标签搜索
java虚拟机
JVM
保姆级教程
Java
Spring AI
SpringBoot
Spring
WebFlux
Nginx
Spring Retry
EasyExcel
流式输出
WebSocket
JustAuth
sso
google
单点登录
源码解析
Tool
图片导出
Luca Ju
累计撰写
39
篇文章
累计收到
1
条评论
首页
栏目
Java 核心
框架与中间件
数据库技术
开发工具与效率
问题排查与踩坑记录
程序员成长与思考
前端
页面
关于这个博客
搜索到
1
篇与
的结果
2026-01-14
为什么不建议使用Executors创建线程池?
在Java开发中,线程池是优化并发性能的核心工具,但线程池的创建方式却藏着不少坑。《阿里巴巴Java开发手册》明确规定:线程池不允许使用Executors创建,必须通过ThreadPoolExecutor手动创建。很多新手可能会疑惑:Executors提供的方法简洁又方便,为什么会被禁止?今天就从底层实现出发,彻底讲清楚这个问题,同时补充线程池的核心知识,帮你避开面试和开发中的高频陷阱。一、先认识下「背锅侠」:Executors类Executors是JUC(java.util.concurrent)包下的工具类,专门用于快速创建线程池,提供了4个核心方法:newFixedThreadPool:固定线程数的线程池newSingleThreadExecutor:单线程线程池newCachedThreadPool:可缓存的线程池newScheduledThreadPool:支持定时/周期性任务的线程池这些方法看似「开箱即用」,但底层参数配置存在致命缺陷,我们逐个拆解。1. 隐患1:newFixedThreadPool & newSingleThreadExecutor——内存溢出风险先看newFixedThreadPool的底层实现代码:public static ExecutorService newFixedThreadPool(int nThreads, ThreadFactory threadFactory) { return new ThreadPoolExecutor(nThreads, nThreads, 0L, TimeUnit.MILLISECONDS, new LinkedBlockingQueue<Runnable>(), threadFactory); }核心问题出在等待队列LinkedBlockingQueue上。我们点进LinkedBlockingQueue的无参构造:public LinkedBlockingQueue() { this(Integer.MAX_VALUE); // 队列长度默认是Integer.MAX_VALUE }关键坑点:Integer.MAX_VALUE是2147483647,相当于「无界队列」。当任务提交速度远大于线程处理速度时,任务会不断堆积在队列中,导致JVM内存持续飙升,最终触发OOM(内存溢出)。newSingleThreadExecutor的问题和它完全一致,底层也是用了无界的LinkedBlockingQueue,且核心线程数固定为1,任务堆积的风险更高:public static ExecutorService newSingleThreadExecutor() { return new FinalizableDelegatedExecutorService (new ThreadPoolExecutor(1, 1, 0L, TimeUnit.MILLISECONDS, new LinkedBlockingQueue<Runnable>())); }2. 隐患2:newCachedThreadPool & newScheduledThreadPool——资源耗尽风险再看newCachedThreadPool的实现:public static ExecutorService newCachedThreadPool() { return new ThreadPoolExecutor(0, Integer.MAX_VALUE, 60L, TimeUnit.SECONDS, new SynchronousQueue<Runnable>()); }核心问题有两个:核心线程数为0:没有常驻线程,每次有任务都需要创建新线程(除非有空闲线程可复用)最大线程数为Integer.MAX_VALUE:理论上可以创建无限多线程关键坑点:当短时间内提交大量任务时,线程池会疯狂创建新线程,而每个线程都会占用一定的内存和CPU资源,最终导致系统资源耗尽,程序崩溃。newScheduledThreadPool的问题类似,最大线程数同样是Integer.MAX_VALUE,存在相同的资源耗尽风险:public static ScheduledExecutorService newScheduledThreadPool(int corePoolSize) { return new ScheduledThreadPoolExecutor(corePoolSize); } // 父类构造(ScheduledThreadPoolExecutor继承自ThreadPoolExecutor) public ScheduledThreadPoolExecutor(int corePoolSize) { super(corePoolSize, Integer.MAX_VALUE, DEFAULT_KEEPALIVE_MILLIS, MILLISECONDS, new DelayedWorkQueue()); }二、正确姿势:ThreadPoolExecutor手动创建(推荐)Executors的问题本质是「参数固化」,无法根据业务场景灵活配置。而ThreadPoolExecutor允许我们手动指定所有核心参数,从根源上避免上述隐患。1. 核心参数详解(面试高频考点)ThreadPoolExecutor的核心构造方法:public ThreadPoolExecutor(int corePoolSize, int maximumPoolSize, long keepAliveTime, TimeUnit unit, BlockingQueue<Runnable> workQueue, ThreadFactory threadFactory, RejectedExecutionHandler handler) { // ... 省略参数校验逻辑 }7个参数的作用必须记牢,直接关系到线程池的性能和稳定性:corePoolSize(核心线程数):线程池的常驻线程数,即使空闲也不会销毁(除非设置allowCoreThreadTimeOut=true)。maximumPoolSize(最大线程数):线程池允许创建的最大线程数,超出核心线程数的是「非核心线程」。keepAliveTime(空闲线程存活时间):非核心线程空闲后的最大存活时间,超时会被销毁,释放资源。unit(时间单位):keepAliveTime的时间单位,如MILLISECONDS(毫秒)、SECONDS(秒)。workQueue(工作队列):用于存放等待执行的任务,必须使用「有界队列」(如ArrayBlockingQueue),避免任务堆积。threadFactory(线程工厂):用于创建线程,可自定义线程名称(方便问题排查)、设置线程优先级等。handler(拒绝策略):当线程数达最大且队列满时,新任务的处理策略(如丢弃任务、抛出异常、由提交线程执行等)。2. 推荐实践:自定义线程池示例结合业务场景(如处理用户订单任务),手动创建线程池:import java.util.concurrent.*; public class ThreadPoolDemo { // 线程工厂:自定义线程名称,方便排查问题 private static final ThreadFactory THREAD_FACTORY = new ThreadFactory() { private int count = 1; @Override public Thread newThread(Runnable r) { Thread thread = new Thread(r); thread.setName("order-thread-pool-" + count++); return thread; } }; // 拒绝策略:队列满时抛出异常,及时发现问题 private static final RejectedExecutionHandler REJECTED_HANDLER = new ThreadPoolExecutor.AbortPolicy(); // 自定义线程池 public static final ThreadPoolExecutor ORDER_THREAD_POOL = new ThreadPoolExecutor( 5, // 核心线程数:根据CPU核心数或业务量配置 10, // 最大线程数:不超过CPU核心数*2(IO密集型可适当增加) 60, // 空闲线程存活时间:60秒 TimeUnit.SECONDS, new ArrayBlockingQueue<>(100), // 有界队列:容量100,避免任务堆积 THREAD_FACTORY, REJECTED_HANDLER ); public static void main(String[] args) { // 提交任务 for (int i = 0; i < 100; i++) { int finalI = i; ORDER_THREAD_POOL.submit(() -> { System.out.println(Thread.currentThread().getName() + " 处理订单:" + finalI); }); } // 关闭线程池(实际项目中可在应用关闭时调用) ORDER_THREAD_POOL.shutdown(); } }三、加餐:线程池工作流程(面试必问)理解线程池的工作流程,能帮你更合理地配置参数。当任务提交后,线程池会按以下步骤处理:提交任务:通过execute()或submit()方法提交任务。检查核心线程:若当前运行的线程数 < corePoolSize,立即创建核心线程执行任务;否则进入下一步。检查工作队列:若队列未满,将任务放入队列等待执行;否则进入下一步。检查最大线程:若当前运行的线程数 < maximumPoolSize,创建非核心线程执行任务;否则进入下一步。触发拒绝策略:线程数达最大且队列满时,执行拒绝策略处理任务。记忆小技巧:核心线程优先接活 → 活太多就放队列 → 队列满了就加临时线程 → 临时线程也满了就拒绝。四、总结Executors被禁止的核心原因是「参数不可控」,导致线程池存在内存溢出或资源耗尽的风险;而ThreadPoolExecutor通过手动配置核心参数,能根据业务场景精准控制线程池的行为,从根源上规避风险。最后再划几个重点:必须使用有界队列(如ArrayBlockingQueue),避免任务堆积。最大线程数需合理配置(CPU密集型:核心数+1;IO密集型:核心数*2)。自定义线程工厂,方便问题排查。选择合适的拒绝策略,避免静默失败。掌握线程池的正确创建方式,不仅能提升程序的稳定性,也是Java面试中的高频考点。希望这篇文章能帮你彻底搞懂这个问题~
2026年01月14日
1 阅读
0 评论
1 点赞