问题 JDK8 CompletableFuture.supplyAsync如何处理interruptedException


CompletableFuture.supplyAsync(
() -> {
    transporter.write(req);
    //here take the value from a blocking queue,will throw a interruptedException
    return responseQueue.take();
},  executorService);

处理interruptedException的常用方法是再次中断或直接抛出interruptedException,但两者都无法工作。有人有想法吗?


2225
2018-04-20 16:42


起源

“但两者都行不通。“=>为什么? - assylias
两者都有编译器错误。如果直接抛出异常,编译器会显示未处理的异常,如果捕获它并调用Thead.current.interrupt,编译器将显示必须返回的T类型。 - GrapeBaBa
是的你需要返回或扔掉。如果您决定返回null,例如: try { return queue.take(); } catch (InterruptedException e) { Thread.currentThread().interrupt(); return null; } - assylias
假设future代表一个正常值或异常的计算结果,我觉得应该是将异常设置为未来的方法,它优于set null。 - GrapeBaBa
我认为lambda函数不支持抛出异常,因此抛出异常是不可能的。你可以做的唯一一件事就是回归。 - mostruash


答案:


我改变了这样的代码。

    CompletableFuture<Rep> result = new CompletableFuture<>();
    CompletableFuture.runAsync(() -> {

        transporter.write(req);
        try {
            Rep rep = responseQueue.take();
            result.complete(rep);
        } catch (InterruptedException e) {
            result.completeExceptionally(e);
            Thread.currentThread().interrupt();
        } catch (Exception e) {
            result.completeExceptionally(e);
        }

    }, executorService);
    return result;

11
2018-04-21 06:07



你所做的就等同于我的。唉, CompletableFuture<Rep> result 可以被符合“结果或异常”范例的任何类替换。例如,您可以添加 get, complete 和 completeExceptionally 方法 ResultWrapper 并使用 ResultWrapper rep = new ResultWrapper();。你使用了这个lambda函数限制是非常巧合的 CompletableFuture 再一次你用它来解决它 CompletableFuture,利用你使用的那些方法。 - mostruash
是的,但是completablefuture已经有了计算结果的抽象,所以我不想为它创建新的类型。 - GrapeBaBa
如果您正在与其他人合作,这可能会使您的代码的读者感到困惑。如果不是,那很好,它对你有用。 - mostruash
那是因为未来并将两个概念转化为java中的一种类型completablefuture。返回一个completablefuture可以使所有调用异步,因为它有很多方法。 - GrapeBaBa
这里没有必要使用CompletableFuture.runAsync,因为你没有使用返回值;如果你想保存一些周期,你可以创建ForJoinTask并在ForkJoinPool#commonPool中执行它。它稍微多一些代码,但是你将避免创建少量实例并写入易失性字段,因此它应该表现得更好。 - Pavel Bucek


由于lambda函数不支持抛出异常,我认为Java开发人员需要一个新的范例。我想到的一件事如下:

public class ResultWrapper<R, E extends Exception> {
    E exception;
    R result;
}

Lambda函数可以返回此包装器的实例。 (编辑:你的情况)

CompletableFuture<ResultWrapper<String, InterruptedException>> aFuture = ...;
...
aFuture.supplyAsync(
() -> {
    try {
        transporter.write(req);
    } catch(InterruptedException e) {
        ResultWrapper<String, InterruptedException> r = new ResultWrapper<>();
        r.exception = e;
        r.result = null;
        return r;
    }
    ...
},  executorService);

3
2018-04-20 17:37



似乎是一个解决方案 - GrapeBaBa
如果您认为它对您有帮助,请将其标记为答案:) - mostruash
您应该将它们包装在未经检查的异常中,并在必要时将它们转换回外部的已检查异常,而不是将异常转换为返回值。将异常作为例外保留在整个过程中。返回值保留用于非错误条件。 - Gili
虽然我在某种程度上同意,但考虑生产者/消费者场景,其中生产/消费对象是lambda函数。消费者必须捕获未经检查的异常,因为生产者可能已抛出未经检查的异常来走私已检查的异常。不确定哪个更糟,我认为这是糟糕的Java。 - mostruash
不要重新发明轮子..为 CompletableFuture.supplyAsync() 把它包起来 java.util.concurrent.CompletionException 并重新抛出它。 - antak


我遇到了同样的问题,但是在阅读了这里的评论和参考书之后,我认为你可以做以下两种中的任何一种:

1(我最终做的):

CompletableFuture.runAsync(() -> {
    transporter.write(req);
    try {
        Rep rep = responseQueue.take();
        result.complete(rep);
    } catch (Exception e) {
        throw new CompletionException(e);
    }
}, executorService);
return result;

或2:

CompletableFuture<Rep> result = new CompletableFuture<>();
new Thread(()-> {
    transporter.write(req);
    try {
        Rep rep = responseQueue.take();
        result.complete(rep);
    } catch (Exception e) {
        retsult.completeExceptionally(e);
    }
}).start();

我知道第二个不使用 executorService,但我觉得使用CompletableFuture的重点是在功能风格中使用CompletionStage API。


2
2018-01-21 23:21



您的第二个解决方案可以改进。看到 stackoverflow.com/a/28961083/868941 详细说明。 - rmuller