Java 8 : Lambda-Streams, 按带有异常的方法过滤

标签 java exception-handling lambda java-8

我在尝试 Java 8 的 Lambda 表达式时遇到问题。 通常它工作正常,但现在我有抛出 IOException 的方法。 最好看看下面的代码:

class Bank{
    ....
    public Set<String> getActiveAccountNumbers() throws IOException {
        Stream<Account> s =  accounts.values().stream();
        s = s.filter(a -> a.isActive());
        Stream<String> ss = s.map(a -> a.getNumber());
        return ss.collect(Collectors.toSet());
    }
    ....
}

interface Account{
    ....
    boolean isActive() throws IOException;
    String getNumber() throws IOException;
    ....
}

问题是,它无法编译,因为我必须捕获 isActive- 和 getNumber-Methods 的可能异常。但即使我明确使用如下所示的 try-catch-Block,它仍然无法编译,因为我没有捕获异常。所以要么JDK有bug,要么我不知道如何捕捉这些异常。

class Bank{
    ....
    //Doesn't compile either
    public Set<String> getActiveAccountNumbers() throws IOException {
        try{
            Stream<Account> s =  accounts.values().stream();
            s = s.filter(a -> a.isActive());
            Stream<String> ss = s.map(a -> a.getNumber());
            return ss.collect(Collectors.toSet());
        }catch(IOException ex){
        }
    }
    ....
}

我怎样才能让它工作?有人可以提示我正确的解决方案吗?

最佳答案

您必须在它转义 lambda 之前 捕获异常:

s = s.filter(a -> {
    try {
        return a.isActive();
    } catch (IOException e) {
        throw new UncheckedIOException(e);
    }
});

请考虑这样一个事实,即 lambda 不是在您编写它的地方评估的,而是在 JDK 类中的某个完全不相关的地方评估的。所以这将是引发检查异常的地方,并且在那个地方它没有被声明。

您可以使用 lambda 的包装器来处理它,该包装器将已检查的异常转换为未检查的异常:

public static <T> T uncheckCall(Callable<T> callable) {
    try {
        return callable.call();
    } catch (RuntimeException e) {
        throw e;
    } catch (Exception e) {
        throw new RuntimeException(e);
    }
}

你的例子可以写成

return s.filter(a -> uncheckCall(a::isActive))
        .map(Account::getNumber)
        .collect(toSet());

在我的项目中,我处理这个问题时不进行包装;相反,我使用一种有效地消除编译器检查异常的方法。不用说,这应该小心处理,项目中的每个人都必须知道,检查的异常可能会出现在未声明的地方。这是管道代码:

public static <T> T uncheckCall(Callable<T> callable) {
    try {
        return callable.call();
    } catch (Exception e) {
        sneakyThrow(e);
        return null; // Unreachable but needed to satisfy compiler
    }
}

public static void uncheckRun(RunnableExc r) {
    try {
        r.run();
    } catch (Exception e) {
        sneakyThrow(e);
    }
}

public interface RunnableExc {
    void run() throws Exception;
}

@SuppressWarnings("unchecked")
private static <T extends Throwable> void sneakyThrow(Throwable t) throws T {
    throw (T) t;
}

你可以期待得到 IOException扔在你的脸上,即使collect没有声明它。在大多数,但不是所有现实生活中,无论如何,您可能只想重新抛出异常,并将其作为一般故障处理。在所有这些情况下,清晰或正确不会丢失任何内容。请注意其他情况,您实际上希望在现场对异常使用react。编译器不会让开发人员知道那里有一个 IOException 要捕获,如果你试图捕获它,编译器实际上会提示,因为我们已经欺骗它相信没有这样的异常可以被扔掉。

关于Java 8 : Lambda-Streams, 按带有异常的方法过滤,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19757300/

相关文章:

java - 如果最小化,Swing 应用程序会崩溃

java - 在java中使用JSON读取和写入字符串文件

python - 在其消费者中处理生成器异常

java - 在 Java 中,使用 throws Exception 而不是抛出多个特定的异常是好的做法吗?

python - 拆分 Pandas 列并将新结果附加到数据框

java - jetty :动态改变空闲时间

java - Type Erasure 抬头,如何规避?

c++ - 为什么 catch block 不共享 try block 的范围?

c++ - lambda 如何捕获自身以进行异步调用?

python - 使用 Lambdas 在 Python 中比较函数