
每次 new Thread() 启动任务,都会创建新线程、分配栈内存、触发上下文切换——资源开销大,且无法复用或统一控制。Executor 框架的核心价值不是“多线程”,而是“可控的并发资源调度”。它把“执行任务”和“管理线程生命周期”解耦,让开发者只关注 Runnable 或 Callable 逻辑。
Java 提供的静态工厂方法看似方便,但不同场景下行为差异极大:
Executors.newFixedThreadPool(n):适合负载稳定、任务耗时相近的场景;但其内部 LinkedBlockingQueue 默认无界,任务积压会 OOMExecutors.newCachedThreadPool():适合大量短生命周期任务;但空闲线程 60 秒后才回收,突发流量可能创建过多线程,压垮系统Executors.newSingleThreadExecutor():保证顺序执行,但单点故障风险高,且无法扩展new ThreadPoolExecutor(...) 手动构造:能精确控制核心线程数、最大线程数、队列类型(如 ArrayBlockingQueue 限容)、拒绝策略(如 AbortPolicy 或自定义)关键区别不在“能不能扔任务”,而在“要不要结果”和“异常处理方式”:
execute(Runnable):只接受 Runnable,无返回值,若任务抛出未捕获异常,该异常会直接打印到 stderr,且线程可能静默终止submit(Runnable) 或 submit(Callable):返回 Future,可调用 get() 获取结果或捕获异常;即使任务抛异常,也封装在 ExecutionException 中,不会丢失submit() + 显式 future.get() 或 future.isDone() 检查不正确关闭会导致任务丢失、线程泄漏或 JVM 无法退出:
立即学习“Java免费学习笔记(深入)”;
shutdown():平滑关闭,不再接收新任务,但会等已提交任务(包括队列中等待的)执行完;需配合 awaitTermination() 等待完成,否则可能提前退出shutdownNow():尝试中断所有正在运行的线程,并清空队列返回未执行任务列表;但中断只是“建议”,任务本身必须响应 Thread.interrupted() 或检查 isInterrupted() 才会真正停止shutdown() 就不管了;或在 finally 块里没加 awaitTermination(30, TimeUnit.SECONDS) 导致主线程结束而子线程还在跑真正安全的关闭流程往往要三步:调 shutdown() → awaitTermination() 等待 → 超时后调 shutdownNow() 强制清理。