如何在Java中设置Timer?

Ank*_*ta 141 java timer

如果连接有任何问题,如何设置一个Timer,比如2分钟,尝试连接数据库,然后抛出异常?

and*_*wmu 267

所以答案的第一部分是如何做主题所要求的内容,因为这是我最初的解释方式,而且有些人似乎觉得有帮助.问题是澄清了,我已经扩展了解决这个问题的答案.

设置计时器

首先你需要创建一个Timer(我在java.util这里使用的是版本):

import java.util.Timer;
Run Code Online (Sandbox Code Playgroud)

..

Timer timer = new Timer();
Run Code Online (Sandbox Code Playgroud)

要执行以下任务,请执行以下操作:

timer.schedule(new TimerTask() {
  @Override
  public void run() {
    // Your database code here
  }
}, 2*60*1000);
// Since Java-8
timer.schedule(() -> /* your database code here */, 2*60*1000);
Run Code Online (Sandbox Code Playgroud)

要在完成持续时间后重复任务:

timer.scheduleAtFixedRate(new TimerTask() {
  @Override
  public void run() {
    // Your database code here
  }
}, 2*60*1000, 2*60*1000);

// Since Java-8
timer.scheduleAtFixedRate(() -> /* your database code here */, 2*60*1000, 2*60*1000);
Run Code Online (Sandbox Code Playgroud)

使任务超时

要专门执行澄清问题所要求的,即尝试在给定时间段内执行任务,您可以执行以下操作:

ExecutorService service = Executors.newSingleThreadExecutor();

try {
    Runnable r = new Runnable() {
        @Override
        public void run() {
            // Database task
        }
    };

    Future<?> f = service.submit(r);

    f.get(2, TimeUnit.MINUTES);     // attempt the task for two minutes
}
catch (final InterruptedException e) {
    // The thread was interrupted during sleep, wait or join
}
catch (final TimeoutException e) {
    // Took too long!
}
catch (final ExecutionException e) {
    // An exception from within the Runnable task
}
finally {
    service.shutdown();
}
Run Code Online (Sandbox Code Playgroud)

如果任务在2分钟内完成,这将在异常情况下正常执行.如果它运行的时间超过了那个,则抛出TimeoutException.

一个问题是虽然你会在两分钟后得到一个TimeoutException,但该任务实际上会继续运行,尽管可能是数据库或网络连接最终会超时并在线程中引发异常.但请注意,在此情况发生之前,它可能会消耗资源

  • 读者小心:这个答案显然是错误的.它将计时器设置为等待2分钟,然后在该2分钟延迟后运行数据库查询,然后每隔2分钟再次运行它.它不做的是立即运行数据库查询,然后在2分钟后失败,这是我认为问题所要求的(在评论中澄清). (7认同)
  • 如何将timer.schedule()与lambda表达式一起使用(从java 8开始)?TimerTask(第一个参数)甚至不是函数式接口。TimerTask只是一个抽象类。 (4认同)
  • TimerTask 不是一个函数式接口!/sf/ask/2657947771/ (3认同)
  • @AgilePro那是真的.我没有回答这个问题,因为它最终已经说明了,我现在更新它以解决这个问题. (2认同)

小智 22

用这个

long startTime = System.currentTimeMillis();
long elapsedTime = 0L.

while (elapsedTime < 2*60*1000) {
    //perform db poll/check
    elapsedTime = (new Date()).getTime() - startTime;
}

//Throw your exception
Run Code Online (Sandbox Code Playgroud)

  • 这实际上是非常糟糕的编码风格,因为while循环不断运行并检查东西......不利于cpu而不利于电池续航时间. (9认同)

and*_*wmu 10

好的,我想我现在明白你的问题.您可以使用Future尝试执行某些操作,然后在发生任何事情后稍微超时.

例如:

FutureTask<Void> task = new FutureTask<Void>(new Callable<Void>() {
  @Override
  public Void call() throws Exception {
    // Do DB stuff
    return null;
  }
});

Executor executor = Executors.newSingleThreadScheduledExecutor();
executor.execute(task);

try {
  task.get(5, TimeUnit.SECONDS);
}
catch(Exception ex) {
  // Handle your exception
}
Run Code Online (Sandbox Code Playgroud)

  • 您可以使用ExecutorService类而不是Executor.它有shutdown()方法来停止执行程序. (2认同)