Java如何实现任务超时处理

简介:

任务超时处理是比较常见的需求,比如在进行一些比较耗时的操作(如网络请求)或者在占用一些比较宝贵的资源(如数据库连接)时,我们通常需要给这些操作设置一个超时时间,当执行时长超过设置的阈值的时候,就终止操作并回收资源。Java中对超时任务的处理有两种方式:一种是基于异步任务结果的超时获取,一种则是使用延时任务来终止超时操作。下文将详细说明。

一、基于异步任务结果的超时获取

基于异步任务结果的获取通常是跟线程池一起使用的,我们向线程池提交任务时会返回一个Future对象,在调用Future的get方法时,可以设置一个超时时间,如果超过设置的时间任务还没结束,就抛出异常。接下来看代码:

 

public class FutureDemo {
static ExecutorService executorService= Executors.newFixedThreadPool(Runtime.getRuntime().availableProcessors()*2);
public static void main(String[] args) { Future<String> future = executorService.submit(new Callable<String>() {
System.out.println("任务被中断。");
@Override public String call() { try { TimeUnit.SECONDS.sleep(10); } catch (InterruptedException e) { } return "OK"; }
} catch (InterruptedException |ExecutionException | TimeoutException e) {
}); try { String result = future.get(2, TimeUnit.SECONDS); future.cancel(true); System.out.println("任务超时。"); }finally {
}
System.out.println("清理资源。"); }
}

运行代码,输出如下:

cc72cb23979165806493a041d3bf24dcdf41ba49

二、使用延时任务来终止超时操作

还有一种实现任务超时处理的思路是在提交任务之前先设置一个定时器,这个定时器会在设置的时间间隔之后去取消任务。当然如果任务在规定的时间内完成了,要记得取消定时器。首先来看一下我们的工作线程:

 

public class RunningTask {
private volatile boolean isStop;
public void stop(){
public void doing() {
this.isStop=true; } int i=1;
TimeUnit.SECONDS.sleep(1);
while (!isStop){ try {
System.out.println("任务被中断。");
} catch (InterruptedException e) { } } }
}

这个工作线程每隔一秒钟会去检查下isStop变量,因此我们可以通过isStop变量来取消任务。至于取消任务的逻辑我们放在了定时器里面,代码如下:

 

public class CancelTask implements Runnable {
private RunningTask runningTask;
this.runningTask = runningTask;
public CancelTask(RunningTask runningTask) { } @Override public void run() {
}
runningTask.stop();
}

可以看到,该定时器的作用就是在一定的时间之后去中断工作线程的运行。接下来测试一下:

 

public class ScheduleDemo {
static ScheduledExecutorService executorService= Executors.newScheduledThreadPool(Runtime.getRuntime().availableProcessors()*2);
public static void main(String[] args) { RunningTask runningTask=new RunningTask();
runningTask.doing();
ScheduledFuture<?> scheduledFuture = executorService.schedule(new CancelTask(runningTask), 3, TimeUnit.SECONDS); if(!scheduledFuture.isDone()){ scheduledFuture.cancel(true); } }
}

运行结果如下:

6997ca65f27dcca622b9098e55610aba59a4e52f

可以看到,任务在超时之后也可以被取消。


原文发布时间为:2018-11-3

本文作者:不才黄某

本文来自云栖社区合作伙伴“Java架构沉思录”,了解相关信息可以关注“Java架构沉思录”。

相关文章
|
7月前
JavaWeb用户信息管理系统-解决HttpSession超时销毁时的异常问题
JavaWeb用户信息管理系统-解决HttpSession超时销毁时的异常问题
31 0
|
6月前
|
人工智能 Java 测试技术
地址标准化通过Java sdk接入通用经纬度查询之前已经调试成功,近期一直返回超时错误
地址标准化通过Java sdk接入通用经纬度查询之前已经调试成功,近期一直返回超时错误
46 1
|
Java
java使用Quartz任务调用crontab表达式的时候报错:Based on configured schedule, the given trigger will never fire
java使用Quartz任务调用crontab表达式的时候报错:Based on configured schedule, the given trigger will never fire
396 0
java使用Quartz任务调用crontab表达式的时候报错:Based on configured schedule, the given trigger will never fire
|
2月前
|
Java 流计算
这个错误信息来自于Java的ScheduledThreadPoolExecutor类,具体的原因是RejectedExecutionException,也就是任务被拒绝执行
【2月更文挑战第3天】这个错误信息来自于Java的ScheduledThreadPoolExecutor类,具体的原因是RejectedExecutionException,也就是任务被拒绝执行
34 2
|
1月前
|
Java 关系型数据库 MySQL
Flink1.18.1和CDC2.4.1 本地没问题 提交任务到服务器 报错java.lang.NoClassDefFoundError: Could not initialize class io.debezium.connector.mysql.MySqlConnectorConfig
【2月更文挑战第33天】Flink1.18.1和CDC2.4.1 本地没问题 提交任务到服务器 报错java.lang.NoClassDefFoundError: Could not initialize class io.debezium.connector.mysql.MySqlConnectorConfig
52 2
|
1月前
|
Java 调度
用Java代码打造每日任务系统
用Java代码打造每日任务系统
22 0
|
2月前
|
Java 流计算
在Flink实时任务中,POJO(Plain Old Java Object)对象的模式演进可能会引起不兼容的问题
【2月更文挑战第6天】在Flink实时任务中,POJO(Plain Old Java Object)对象的模式演进可能会引起不兼容的问题
22 3
|
7月前
|
安全 Java
深入理解Java单例模式和优化多线程任务处理
深入理解Java单例模式和优化多线程任务处理
|
8月前
|
存储 缓存 Java
Java线程池-当任务渐增时的处理-各个参数的含义
Java线程池-当任务渐增时的处理-各个参数的含义
46 0
|
4月前
|
Java
Java【代码分享 10】线程池ThreadPoolExecutor指定线程执行任务(修改线程名称+线程任务指定)
Java【代码分享 10】线程池ThreadPoolExecutor指定线程执行任务(修改线程名称+线程任务指定)
80 0