问题
通过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();
}
}
}
public static void main(String[] args) {
ScheduledExecutorService service
= Executors.newScheduledThreadPool(1);
service.scheduleAtFixedRate(
new Runner(), 0, 1, TimeUnit.SECONDS);
}