Java线程池停止超时任务

问题

通过ScheduledExecutorService线程池定时调度一个任务,如果其中一次调度任务卡住的话,不仅这次调度失败,而且整个线程池也会停在这次调度上。

解决思路

Future

定时线程池没有任务调度的超时策略,但是Future有。

Future(java.util.concurrent Interface Future<V>)表示异步计算的结果。Future接口提供了检查计算是否完成、检查计算是否被取消、等待计算完成并获取计算结果等方法。可以结合Future,提供一种自动停止超时任务的方式,来解决某个任务超时的问题。

用到的方法:

get(long timeout, TimeUnit unit)

如果在指定时间内没有完成计算,则会抛出 TimeoutException。

cancel(boolean mayInterruptIfRunning)

取消当前线程的执行。mayInterruptIfRunning参数表示是否在线程执行的过程中阻断。

private static class Caller implements Callable<Boolean> {
        @Override
        public Boolean call() {
            try {
                Thread.sleep(10000);
                System.out.println(new Date());
                return true;
            } catch (Exception e) {
                e.printStackTrace();
            return false;
    private static class Runner implements Runnable {
        @Override
        public void run() {
            ExecutorService excutor = Executors.newSingleThreadExecutor();
            Future<Boolean> future = excutor.submit(new Caller());
            try {
                future.get(1, TimeUnit.SECONDS);
            } catch (TimeoutException e) {
                System.out.println("timeout");
                future.cancel(true);
            } catch (Exception e) {
                e.printStackTrace();