2

在“可选”源代码中,我发现了这个函数:

public <X extends Throwable> T orElseThrow(Supplier<? extends X> exceptionSupplier) throws X {
    if (value != null) {
        return value;
    } else {
        throw exceptionSupplier.get();
    }
}

我的问题是,如果我将功能更改为此,它看起来像工作一样

public <X extends Throwable> T orElseThrow(Supplier<X> exceptionSupplier) throws X {
    if (value != null) {
        return value;
    } else {
        throw exceptionSupplier.get();
    }
}

有人知道原因吗?

4

1 回答 1

4

考虑这种方法:

void example() throws IOException {
  throw new FileNotFoundException();
}

这声明它抛出了一般的IOException,但具体地抛出了一个更具体的异常,FileNotFoundException

orElseThrow: 通过接受上限供应商相同,它可以抛出更具体的异常类型。

在大多数情况下,差异是无关紧要的,因为您总是可以捕获/throws更通用的异常类型。我认为它可能会有所不同的一种情况是当您接受Supplier作为参数时:

<X extends IOException> void example(Supplier<? extends X> exceptionSupplier)
    throws IOException {  // Note IOException, not X.
  Optional.empty().orElseThrow(exceptionSupplier);
}

您可以使用以下任一供应商作为参数调用它:

Supplier<IOException> s1 = IOException::new;
Supplier<FileNotFoundIOException> s2 = FileNotFoundIOException::new;

但是如果没有上界,你就不能做后者Supplier<? extends X>

于 2021-04-21T08:28:40.253 回答