设置方法/线程的最长执行时间

2022-09-04 08:01:00

我有一个方法,它写入数据库。要求是确保此方法在经过特定时间后不执行。如果它在此之前返回,那么什么都不应该做。

我能想到的一个基本方法是做这样的事情。

public class LimitedRuntime {

    public static void writeToDb(){
            // writes to the database
    }

    public static void main(String[] args) {
        long totalExecutionTime = 8000L;
        long startTime = System.currentTimeMillis();

        while(System.currentTimeMillis() - startTime < totalExecutionTime )
        {
            writeToDb();
        }   
    }
}

此方法的一个问题是,即使该方法在最大总执行时间之前返回,即使这样,程序也会停止以等待时间过去。

我怎样才能做得更好(或者更正确)?如果我们使用 ,我们怎么能找出哪个执行该方法?ThreadThread


答案 1

您可以通过将作业发送给执行程序来执行此操作:

 public static void main(String[] args) {
    ExecutorService executor = Executors.newFixedThreadPool(4);

    Future<?> future = executor.submit(new Runnable() {
        @Override
        public void run() {
            writeToDb();            //        <-- your job
        }
    });

    executor.shutdown();            //        <-- reject all further submissions

    try {
        future.get(8, TimeUnit.SECONDS);  //     <-- wait 8 seconds to finish
    } catch (InterruptedException e) {    //     <-- possible error cases
        System.out.println("job was interrupted");
    } catch (ExecutionException e) {
        System.out.println("caught exception: " + e.getCause());
    } catch (TimeoutException e) {
        future.cancel(true);              //     <-- interrupt the job
        System.out.println("timeout");
    }

    // wait all unfinished tasks for 2 sec
    if(!executor.awaitTermination(2, TimeUnit.SECONDS)){
        // force them to quit by interrupting
        executor.shutdownNow();
    }
}

答案 2

还有一个AspectJ解决方案,其中包含jcabi方面库:

@Timeable(limit = 5, unit = TimeUnit.SECONDS)
public String writeToDb() {
  // writeToDb
}

有一篇文章进一步解释了它:限制Java方法执行时间


推荐