当前位置: 首页 > 面试题库 >

Java 8:Lambda-Streams,按方法进行过滤(异常)

林华皓
2023-03-14
问题内容

在尝试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,它仍然无法编译,因为我没有捕获到Exception。因此,要么JDK中存在错误,要么我不知道如何捕获这些异常。

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

考虑以下事实:在JDK类中,lambda不在你编写的位置评估,而是在某些完全不相关的位置评估。这样便可以抛出该检查异常,并且在该位置不声明该异常。

你可以通过使用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) { return sneakyThrow(e); }
}
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没有声明,你也可能会被扔到脸上。在大多数(但不是全部)实际情况下,无论如何,你都只想抛出异常,并将其作为一般故障进行处理。在所有这些情况下,清晰度或正确性都不会丢失。只是要提防那些其他情况,你实际上想在现场对异常做出反应。编译器不会使开发人员意识到存在一个IOException陷阱,实际上,如果你尝试捕获它,编译器实际上会抱怨,因为我们欺骗了它以为不会抛出此类异常。



 类似资料:
  • 我希望得到以下内容:,然后:。因为我们首先遍历1到20之间的所有数字,所以只筛选出那些可被3除的数字,然后遍历这个新的并找到那些可被5除的数字。 但相反,我得到了以下输出: 它看起来没有检查所有的数字。此外,它看起来只检查可被3除的数字。

  • 问题内容: 我想知道是否有人可以向我解释以下怪异现象。我正在使用Java 8 Update 11。 给定这种方法 如果我首先构造一个函数Object,并将其传递给上面的方法,那么事情就会编译。 但是,如果我以lambda形式内联函数,则编译器会说 未报告的异常X; 必须被抓住或宣布被抛出 更新 :原来错误消息由maven缩写。如果直接使用javac编译,则错误为: 另请参阅此处以获取可运行的测试代

  • 我想知道是否有人能向我解释以下奇怪之处。我正在使用Java8更新11。 给定此方法 如果我首先构造一个函数对象,并将其传递给上面的方法,事情就会编译。 但是,如果我将函数作为lambda内联,编译器会说 未报告的异常X;必须被抓到或宣布被抛出 更新:原来错误消息是由maven缩写的。如果直接使用javac编译,错误是: 另请参阅此处以获取可运行的测试代码。

  • 问题内容: 我有两个列表,我要过滤两个列表都包含的thoose元素。我想用lambda表达式做到这一点。 用户getName和客户端getUserName都返回String。 这是我的示例代码: 问题答案: 但这效率很低,因为它是O(m * n)。您最好创建一组可接受的名称: 还要注意,它并不严格等同于您拥有的代码(如果已编译),如果有多个用户具有与该客户端相同的名称,则该客户端会将同一客户端两次

  • 我有这个列表,我想按保留顺序订购,但我没有在自动完成辅助中找到任何函数 我也尝试过: 但是我有编译错误:

  • 我有一个对象列表,比如。我想根据使用Java 8的一些参数过滤这个列表。但如果参数为,则抛出。如何过滤掉空值? 当前代码如下 如果返回,则抛出。