Java 8:Lambda 流和異常處理
在 Java 8 中,lambda 表達式為流處理帶來了許多好處。然而,在處理拋出檢查異常的方法時,例如 IOException,開發人員可能會遇到編譯錯誤。
考慮以下程式碼:
<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()); } }</code>
程式碼旨在擷取一組已啟動的帳號。但是,它無法編譯,因為 isActive 和 getNumber 方法拋出 IOExceptions,必須在簽章中擷取或宣告該例外。
捕獲 Lambda 表達式中的已檢查異常
To要解決此問題,必須在 lambda 表達式本身內處理已檢查的異常。然而,僅僅在 lambda 中放置一個 try-catch 區塊是不夠的。必須在異常逃逸 lambda 作用域之前捕捉該異常。
使用 UncheckedIOException 包裝器
一種方法是使用自訂包裝器類別 UncheckedIOException 來轉換檢查的異常到未經檢查的。這使得 lambda 能夠拋出未經檢查的異常並由流的後續操作處理。
<code class="java">s = s.filter(a -> { try { return a.isActive(); } catch (IOException e) { throw new UncheckedIOException(e); } });</code>
使用通用異常包裝器
另一種選擇是使用通用包裝方法uncheckCall,捕獲任何類型的異常並將其作為未經檢查的異常重新拋出。
<code class="java">return s.filter(a -> uncheckCall(a::isActive)) .map(Account::getNumber) .collect(toSet());</code>
在這種情況下,流的後續操作將收到未經檢查的異常並可以進行相應的處理。
解除編譯器的異常檢查
更高級的方法涉及使用有效禁用編譯器異常檢查的方法。然而,這需要謹慎並清楚地了解潛在風險。在此方法中,使用以下程式碼:
<code class="java">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 } }</code>
使用此方法,將捕獲已檢查的異常並將其作為運行時異常重新拋出。但是,需要注意的是,如果要在 lambda 執行網站處理異常,則此方法可能會導致意外行為。
以上是如何處理 Java 8 Lambda 流中的檢查異常?的詳細內容。更多資訊請關注PHP中文網其他相關文章!