Callable用于让线程执行任务后返回结果并抛出受检异常,弥补Runnable无返回值和无法声明受检异常的缺陷;需配合ExecutorService和Future使用,通过Future.get()获取结果或异常。

Callable 接口用来让线程执行任务后返回结果,弥补了 Runnable 不能返回值、不能抛出受检异常的限制。
为什么需要 Callable
Runnable 是最常用的线程任务接口,但它有两个明显短板:
- run() 方法返回 void,无法直接获取执行结果
- run() 不能声明抛出受检异常(比如 IOException),异常只能在内部 try-catch,不利于错误传递
当你要“启动一个线程做点事,然后拿回计算结果”时(比如远程调用、耗时计算、数据库查询),Runnable 就不够用了。Callable 正是为此而生。
Callable 的核心特点
Callable 是一个泛型接口,定义如下:
立即学习“Java免费学习笔记(深入)”;
public interface Callable{ V call() throws Exception; }
- call() 方法有返回值(类型由泛型 V 决定)
- call() 可以抛出任意异常(包括受检异常)
- 它本身不能直接交给 Thread 启动,必须配合 ExecutorService 使用
怎么用 Callable 获取返回值
关键靠 Future —— 它是任务的“占位符”,代表异步计算的结果:
- 提交 Callable 到线程池,返回一个 Future 对象
- Future.get() 会阻塞等待结果(可设超时),拿到 call() 的返回值
- 如果 call() 抛出异常,get() 会包装成 ExecutionException 抛出
示例片段:
ExecutorService pool = Executors.newFixedThreadPool(2); Callabletask = () -> { Thread.sleep(1000); return 42; // 返回计算结果 }; Future future = pool.submit(task); Integer result = future.get(); // 阻塞等待,得到 42 pool.shutdown();
和 Runnable 的对比与选择
不是替代,而是分工:
- 只关心“执行动作”,不关心结果 → 用 Runnable(简单、轻量)
- 需要结果、要处理异常、想统一管理任务生命周期 → 用 Callable + Future
- 如果还想简化回调或链式处理,可进一步升级到 CompletableFuture
注意:Future.get() 默认阻塞,若不想卡住主线程,可用 isDone() 轮询,或结合超时参数使用。










