Java 8 Lambda-Streams: Filtering by Method with Exceptions
Intercepting exceptions thrown from methods invoked within lambda expressions presents a challenge when utilizing the enhanced Java 8 stream processing capabilities. The issue arises when these methods declare checked exceptions, which the enclosing lambda is not permitted to throw.
Consider the following example:
<code class="java">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; }</code>
To compile this code, it is necessary to catch the potential IOException in the isActive and getNumber methods. However, merely handling the exceptions within a try-catch block, as demonstrated below, still results in a compilation error:
<code class="java">class Bank { 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) { // Exception not caught } } }</code>
To resolve this issue, the exception must be captured before it escapes the lambda expression. This can be achieved by wrapping the lambda in a custom function that translates the checked exceptions into unchecked ones:
<code class="java">s = s.filter(a -> { try { return a.isActive(); } catch (IOException e) { throw new UncheckedIOException(e); // Translated to an unchecked exception } });</code>
Alternatively, an approach that avoids wrapping can be employed:
<code class="java">public static <T> T uncheckCall(Callable<T> callable) { try { return callable.call(); } catch (Exception e) { sneakyThrow(e); // Potentially throws the exception return null; // Unreachable, but necessary to satisfy the compiler } }</code>
This function essentially tricks the compiler into believing that no checked exceptions can be thrown, allowing the exception to be handled gracefully at a higher level.
<code class="java">return s.filter(a -> uncheckCall(a::isActive)) .map(Account::getNumber) .collect(toSet());</code>
By applying these techniques, it is possible to utilize lambda expressions with methods that declare checked exceptions, ensuring both code clarity and exception handling capabilities.
The above is the detailed content of How to Handle Checked Exceptions Thrown from Methods Invoked Within Java 8 Lambda Expressions?. For more information, please follow other related articles on the PHP Chinese website!