电话
400 9058 355
Future.get() 不阻塞的根本原因是任务未启动或线程池已拒绝任务;shutdown后submit直接抛RejectedExecutionException,不返回Future。
根本原因在于任务本身是否真正启动或是否被线程池接受。如果

ExecutorService 已 shutdown() 或 shutdownNow(),后续 submit() 会直接抛出 RejectedExecutionException,根本不会返回 Future;而即使返回了 Future,若任务因异常在构造阶段失败(如 Callable 的 call() 方法抛出未捕获异常),get() 仍会阻塞——但最终抛出 ExecutionException 包裹原始异常。
实操建议:
submit() 前确认线程池状态:if (!executor.isShutdown() && !executor.isTerminated()) { ... }
Future.get() 加超时控制,避免无限等待:future.get(3, TimeUnit.SECONDS)
ExecutionException、TimeoutException、CancellationException 三类异常,不可只 catch Exception
核心差异在返回值与异常传播机制。Runnable 无返回值、run() 方法不能抛受检异常;Callable 有泛型返回值、call() 可抛任意异常,且该异常会被封装进 Future.get() 抛出的 ExecutionException 中。
实操建议:
Callable,别试图在 Runnable 里写全局变量“取结果”submit(Runnable task) 实际返回的是 Future>,其 get() 永远返回 null,不是“没结果”,而是设计如此Lambda 写 Callable,注意类型推导:executor.submit(() -> { doWork(); return "done"; }) 的返回类型由 Lambda 主体决定没有通用值,取决于任务 I/O 特性与 CPU 密集度。设错会导致线程饥饿或资源浪费:core 太小,短突发任务排队;max 太大,大量空闲线程争抢 CPU 上下文。
实操建议:
Runtime.getRuntime().availableProcessors(),core = max = N 或 N+1LinkedBlockingQueue(无界)或带容量的 ArrayBlockingQueue,避免拒绝策略误杀Executors.newCachedThreadPool() 生产环境——它的 maximumPoolSize 是 Integer.MAX_VALUE,OOM 风险极高直接调用 shutdown() 不等于“立刻停”,它只是停止接收新任务;已提交任务继续执行。必须配合 awaitTermination() 等待结束,否则 JVM 可能提前退出,导致任务被强行中断。
实操建议:
shutdown() → awaitTermination() → 若超时则 shutdownNow()
awaitTermination() 返回 false 表示超时,此时再调 shutdownNow() 会尝试中断正在运行的任务(依赖任务自身响应 Thread.interrupted())finally 块中执行关闭逻辑,防止异常跳过:try {
executor.submit(() -> doWork());
} finally {
executor.shutdown();
try {
if (!executor.awaitTermination(10, TimeUnit.SECONDS)) {
executor.shutdownNow();
}
} catch (InterruptedException e) {
executor.shutdownNow();
Thread.currentThread().interrupt();
}
}
最易被忽略的一点:任务内部若用了不可中断的阻塞操作(如 InputStream.read()、某些 JNI 调用),shutdownNow() 无法真正终止它,只能靠超时或外部信号。这种任务必须自行实现可取消逻辑。
邮箱:8955556@qq.com
Q Q:8955556
本文详解如何将Go官方present工具(用于生成HTML5...
PySNMP在不同版本中对SNMP错误状态(errorSta...
time.Sleep仅阻塞当前goroutine,其他gor...
PHPfopen()创建含特殊符号的文件名失败主因是操作系统...
WooCommerce中通过代码为分组产品动态聚合子商品的属...
io.ReadFull返回io.ErrUnexpectedE...
本文详解Yii2中控制器向视图传递ActiveRecord数...
本文详解为何通过wp_set_object_terms()为...
Pytest中使用@mock.patch类装饰器会导致补丁泄...
带缓冲的channel是并发安全的FIFO队列;make(c...