FutureTask用于封装异步任务并获取结果,支持取消与状态查询。1. 可包装Callable并交由线程或线程池执行,通过get()阻塞获取结果;2. 结合ExecutorService更高效管理资源;3. 提供isDone、isCancelled、cancel等方法控制任务生命周期;4. get()需处理ExecutionException、InterruptedException及TimeoutException,超时可取消任务。适用于需精确控制单次异步任务的场景。

在Java中,FutureTask 是一个可取消的异步计算任务,它实现了 Future 和 Runnable 接口,可以用来包装 Callable 或 Runnable 对象,便于管理任务的执行状态和获取结果。它常与线程池或独立线程配合使用,实现任务提交与结果获取的分离。
1. FutureTask的基本用法
FutureTask通常用于封装一个有返回值的Callable任务。创建后可以交给线程执行,并通过get()方法获取结果。
示例:Callabletask = () -> { Thread.sleep(2000); return "任务执行完成"; }; FutureTask futureTask = new FutureTask<>(task); // 启动线程执行任务 new Thread(futureTask).start(); // 获取结果(阻塞直到任务完成) try { String result = futureTask.get(); // 可设置超时:get(3, TimeUnit.SECONDS) System.out.println(result); } catch (Exception e) { e.printStackTrace(); }
2. 结合线程池使用FutureTask
虽然可以直接用Thread运行FutureTask,但更常见的做法是将其提交给线程池执行,便于资源管理。
示例:ExecutorService executor = Executors.newSingleThreadExecutor(); CallablecomputeTask = () -> { int sum = 0; for (int i = 1; i <= 100; i++) sum += i; return sum; }; FutureTask future = new FutureTask<>(computeTask); executor.execute(future); try { Integer result = future.get(); System.out.println("计算结果:" + result); } catch (Exception e) { e.printStackTrace(); } finally { executor.shutdown(); }
3. 任务状态控制与取消
FutureTask支持对任务执行过程进行干预,比如检查是否完成、是否被取消,或主动取消任务。
立即学习“Java免费学习笔记(深入)”;
- isDone():判断任务是否已完成(正常结束、异常或被取消)
- isCancelled():判断任务是否已被取消
- cancel(boolean mayInterruptIfRunning):尝试取消任务
FutureTasklongTask = new FutureTask<>(() -> { for (int i = 0; i < 10; i++) { if (Thread.interrupted()) { throw new InterruptedException("任务被中断"); } Thread.sleep(500); } return "执行成功"; }); new Thread(longTask).start(); // 2秒后尝试取消 Thread.sleep(2000); boolean cancelled = longTask.cancel(true); System.out.println("任务是否成功取消:" + cancelled); // 检查状态 if (longTask.isCancelled()) { System.out.println("任务已取消"); } else if (longTask.isDone()) { System.out.println("任务已完成"); }
4. 处理异常与超时
调用 get() 方法可能抛出异常,需合理处理;也可设置超时避免无限等待。
- ExecutionException:任务执行过程中抛出异常
- InterruptedException:当前线程等待结果时被中断
- TimeoutException:get(timeout) 超时时抛出
try {
String result = futureTask.get(1, TimeUnit.SECONDS);
System.out.println(result);
} catch (TimeoutException e) {
System.out.println("任务超时");
futureTask.cancel(true); // 超时后取消任务
} catch (ExecutionException e) {
System.out.println("任务执行出错:" + e.getCause().getMessage());
} catch (InterruptedException e) {
System.out.println("等待结果时被中断");
}
基本上就这些。FutureTask适合需要精确控制任务生命周期、获取异步结果的场景,尤其适用于单次执行且关心结果的任务。结合线程池使用时注意及时释放资源,避免内存泄漏。不复杂但容易忽略细节,比如异常处理和取消机制。










