如何在Java中设置计时器?


151

如何设置一个计时器(例如2分钟)以尝试连接到数据库,然后在连接中出现任何问题时引发异常?


1
Coul OP澄清了他们是否希望简单尝试至少2分钟的操作,或者是否必须立即在两分钟后抛出异常,即使当前正在尝试进行连接
thecoshman,2013年

Answers:


280

因此,答案的第一部分是如何做主题所要求的,因为这是我最初对它的理解方式,并且似乎有人发现有所帮助。此后,问题得到了澄清,我已经扩展了答案以解决该问题。

设定计时器

首先,您需要创建一个计时器(我在java.util这里使用版本):

import java.util.Timer;

..

Timer timer = new Timer();

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

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);

要在持续时间之后重复执行任务,您可以执行以下操作:

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);

使任务超时

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

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();
}

如果任务在2分钟内完成,它将正常执行,但有例外。如果运行时间超过该时间,则将抛出TimeoutException。

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


哦,我想你把我的问题弄错了,我不得不反复尝试连接数据库直到每次2分钟
Ankita 2010年

对不起,但我的要求是我想给2分钟时间连接数据库,然后它只会抛出一些错误消息。这就像我想设置一个尝试连接数据库的时间限制
Ankita

7
读者注意:这个答案显然是错误的。它将计时器设置为等待2分钟,然后在2分钟的延迟后运行数据库查询,然后每2分钟一次又一次运行它。它不执行的操作是立即运行数据库查询,然后在2分钟后失败,这就是我认为该问题所要求的(如注释中所阐明)。
AgilePro

2
@AgilePro是的。我没有回答这个问题,因为最终提出了这个问题,现在我已对其进行更新以解决该问题。
andrewmu

1
@ErnestasGruodis核心API将构造函数公开列出:docs.oracle.com/javase/7/docs/api/java/util/Timer.html#Timer()您的类路径中可能有一个不同的Timer类-请尝试使用Java。 util.Timer作为类。
andrewmu

25

用这个

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

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

//Throw your exception

1
嗯...这在技术上是可行的,但是它不包括某些
极端

1
这是唯一正确的答案。它将在单个线程上运行,并且将计算结束时间而不会产生漂移。在这种情况下,使用TimerTask完全是错误的事情。令我惊讶的是,StackOverflow上有多少示例表明了这种错误观念。
AgilePro 2013年

1
@thecoshman-计时器实现不会在正在进行的DB操作中中断它,也不会这样做。无论如何,您只能控制启动数据库操作的时间。有一个普遍的误解,认为您需要一个“计时器”来对事物进行计时,但是您并不需要。您只需要做某事,并利用当前时间来确保您做的时间不会太长。
AgilePro

假设您必须返回连接或在两分钟后抛出,这将无法正常工作。如果尝试连接两分钟后,您仍然无法连接。在连接检查需要两个星期的情况下,这将花费很长时间才能引发异常。
thecoshman

14
这实际上是非常糟糕的编码样式,因为while循环不断运行并检查内容...对CPU不利,对电池寿命不利。
2013年

11

好的,我想我现在已经了解您的问题了。您可以使用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
}

抛出异常,但程序未终止。请帮助我。
Ankita

2
您可以使用ExecutorService类代替Executor。它具有shutdown()方法来停止执行程序。
仪式

1
执行器将吞下您未在Callable / Runnable中专门处理的抛出异常。如果您的Runnable / Callable本身没有捕获并处理该异常,并且将异常提供给您或您拥有它,那么您需要将ScheduledExecutorService子类化并重写afterExecute(确保调用super.afterExecute())。afterExecute的第二个参数是Runnable / Callable中的throwable
亚当

3
    new java.util.Timer().schedule(new TimerTask(){
        @Override
        public void run() {
            System.out.println("Executed...");
           //your code here 
           //1000*5=5000 mlsec. i.e. 5 seconds. u can change accordngly 
        }
    },1000*5,1000*5); 

只需将上面的代码复制到有害生物即可。我给了两次“时间”,第一次是第一次执行此代码,第二次是间隔时间。
Mahadev Mane'5

Timer的文档建议改为使用Executor框架。docs.oracle.com/javase/10/docs/api/java/util/Timer.html
Karan Khanna,

1

[Android]如果有人希望使用java在android上实现计时器

您需要像这样使用UI线程来执行操作。

Timer timer = new Timer();
timer.schedule(new TimerTask() {
           @Override
            public void run() {
                ActivityName.this.runOnUiThread(new Runnable(){
                    @Override
                      public void run() {
                       // do something
                      }        
                });
            }
        }, 2000));
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.