调用get()方法时会抛出ExecutionException,其getCause()返回原始异常;需捕获ExecutionException并提取cause以获取真实异常信息。

在Java中使用FutureTask时,如果任务执行过程中抛出异常,不能直接看到异常信息,必须通过特定方式获取。这是因为异常被封装在FutureTask内部,只有调用 get() 方法时才会被重新抛出。
异常如何被捕获和封装
FutureTask 执行的任务(如Callable)若抛出异常,该异常会被捕获并存储在任务内部。调用 get() 时,FutureTask 会将原始异常包装为 ExecutionException 抛出。
例如:
FutureTasktask = new FutureTask<>(() -> { throw new RuntimeException("计算失败"); }); new Thread(task).start(); try { Integer result = task.get(); // 此处抛出 ExecutionException } catch (ExecutionException e) { Throwable cause = e.getCause(); // 获取原始异常:RuntimeException System.out.println("实际异常: " + cause.getMessage()); }
正确获取异常的步骤
要准确获取FutureTask中的异常,需注意以下几点:
立即学习“Java免费学习笔记(深入)”;
-
必须调用 get() 方法:只有调用
get()或get(long, TimeUnit)才能触发异常的传递。 -
捕获 ExecutionException:这是任务执行失败的标志,其
getCause()返回真正的异常。 - 区分 InterruptedException:等待过程中线程被中断会抛出此异常,与任务本身异常无关。
完整异常处理示例
try {
Integer result = task.get(5, TimeUnit.SECONDS);
} catch (InterruptedException e) {
Thread.currentThread().interrupt(); // 恢复中断状态
System.err.println("等待结果时被中断");
} catch (ExecutionException e) {
Throwable realException = e.getCause();
System.err.println("任务执行出错: " + realException.getMessage());
} catch (TimeoutException e) {
System.err.println("任务超时未完成");
}
这样可以全面覆盖任务执行中的各种异常情况,尤其是 ExecutionException 的 getCause() 是获取真实错误的关键。
get() 时处理 ExecutionException,并通过 getCause() 提取原始异常,就能准确掌握 FutureTask 中的问题根源。










