java - java中导致超时异常时如何获得Future返回

标签 java future executorservice

我想知道当使用Future和ExecutorService时导致超时异常时我能否得到结果,例如,下面是我的代码

package com.example;

import java.util.UUID;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Future;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.concurrent.ThreadPoolExecutor;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;

import org.apache.commons.lang3.concurrent.BasicThreadFactory;
import org.junit.Test;

public class FutureTest {
    public static String sayHello() {
        try {
            Thread.sleep(3000);
            return "world";
        } catch (Exception e) {
            return "hello";
        }
    }

    @Test
    public void testTimeoutAndInterrupt() throws Exception {
        ExecutorService executorService = new ThreadPoolExecutor(0, Runtime.getRuntime().availableProcessors() * 2, 0,
            TimeUnit.SECONDS, new LinkedBlockingQueue<>(10), new BasicThreadFactory.Builder().namingPattern(
            "executor-pool-%d").daemon(true).build());

        String processName = UUID.randomUUID().toString();

        Future<String> future = executorService.submit(
            () -> {
                return sayHello();
            }
        );
        try {
            String result = future.get(2, TimeUnit.SECONDS);
            System.out.println("future result:" + result);
        } catch (TimeoutException e) {
            System.out.println("timeout exception");
        } finally {
            future.cancel(true);
            System.out.println("cancel by future");
        }
    }
}

当我使用 future.get(5, TimeUnit.SECONDS) 时,可以按照我的预期获取值“world”。 现在,当我使用 future.get(2, TimeUnit.SECONDS) 时,我想要得到的返回值为“hello”。

我该怎么做?谢谢

最佳答案

不,该异常不会在执行过程中引发。相反,会抛出InterruptedException。 (在那里尝试 System.out.println)。

控制流程:

    1. 执行超时
  • 线程被中断
  • 3a。抛出 TimeoutException
  • 3b。 FutureTask 收到 InterruptedException

由于 3a 和 3b 在两个线程中发生(异步),因此顺序可能是 3b;3a。然而 3a;3b 似乎是自然顺序。

这样做:

    String result;
    try {
        result = future.get(2, TimeUnit.SECONDS);
        System.out.println("future result:" + result);
    } catch (TimeoutException e) {
        System.out.println("timeout exception");
        result = "hello";
    } finally {

注意:我认为 FutureTask 线程不能保证关闭(因为可能没有捕获,即使在捕获之后代码仍会继续)。

使用catch InterruptedException在某个地方存入结果,将会失败,因为在捕获TimeoutException的时刻,FutureTask仍然不会有InterruptedException。

在每个循环步骤中存储进度时,这也是一个陷阱。捕获超时后,进度可能仍会更新。

关于java - java中导致超时异常时如何获得Future返回,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/48109628/

相关文章:

java - java中使用线程的定时器

java - 如何检查原始空数组

Java : BufferedImage to Bitmap format

java - Android Java fragment 显示在半屏上

java - future 和 promise 有什么区别?

Flutter/Dart - 加载 future 的 CircularProgressIndicator 的透明背景?

Java Future.isDone 返回 true,即使它不应该返回 true,这会停止程序进程

dart - 我什么时候应该使用 FutureBuilder?

java - ThreadPoolExecutor 仅运行第一个可运行对象

java - 如何立即关闭executorservice中的所有线程?