Java語(yǔ)法中Lambda表達(dá)式無(wú)法拋出異常的解決
Lambda表達(dá)式無(wú)法拋出異常
1.Demo 例子
錯(cuò)誤提示 - Unhandled exception: java.io.IOException;
public static void main(String[] args) throws IOException{ Stream.of("a", "b", "c").forEach(str -> { throw new IOException(); }); }
2.編譯通過(guò)
Stream.of("a", "b", "c").forEach(str -> { throw new RuntimeException(new IOException()); }); // 不建議使用
或者
static <E extends Exception> void doThrow(Exception e) throws E { throw (E)e; } //編譯通過(guò) Stream.of("a", "b", "c").forEach(str -> { doThrow(new IOException()); });
lambda表達(dá)式異常應(yīng)該如何處理
java 8中引入了lambda表達(dá)式,lambda表達(dá)式可以讓我們的代碼更加簡(jiǎn)介,業(yè)務(wù)邏輯更加清晰,但是在lambda表達(dá)式中使用的Functional Interface并沒(méi)有很好的處理異常,因?yàn)镴DK提供的這些Functional Interface通常都是沒(méi)有拋出異常的,這意味著需要我們自己手動(dòng)來(lái)處理異常。
因?yàn)楫惓7譃閁nchecked Exception和checked Exception,我們分別來(lái)討論。
處理Unchecked Exception
Unchecked exception也叫做RuntimeException,出現(xiàn)RuntimeException通常是因?yàn)槲覀兊拇a有問(wèn)題。RuntimeException是不需要被捕獲的。也就是說(shuō)如果有RuntimeException,沒(méi)有捕獲也可以通過(guò)編譯。
我們看一個(gè)例子
List<Integer> integers = Arrays.asList(1,2,3,4,5); integers.forEach(i -> System.out.println(1 / i));
這個(gè)例子是可以編譯成功的,但是上面有一個(gè)問(wèn)題,如果list中有一個(gè)0的話,就會(huì)拋出ArithmeticException。
雖然這個(gè)是一個(gè)Unchecked Exception,但是我們還是想處理一下:
integers.forEach(i -> { try { System.out.println(1 / i); } catch (ArithmeticException e) { System.err.println( "Arithmetic Exception occured : " + e.getMessage()); } });
上面的例子我們使用了try,catch來(lái)處理異常,簡(jiǎn)單但是破壞了lambda表達(dá)式的最佳實(shí)踐。代碼變得臃腫。
我們將try,catch移到一個(gè)wrapper方法中:
static Consumer<Integer> lambdaWrapper(Consumer<Integer> consumer) { return i -> { try { consumer.accept(i); } catch (ArithmeticException e) { System.err.println( "Arithmetic Exception occured : " + e.getMessage()); } }; }
則原來(lái)的調(diào)用變成這樣:
integers.forEach(lambdaWrapper(i -> System.out.println(1 / i)));
但是上面的wrapper固定了捕獲ArithmeticException,我們?cè)賹⑵涓木幊梢粋€(gè)更通用的類(lèi):
static <T, E extends Exception> Consumer<T> consumerWrapperWithExceptionClass(Consumer<T> consumer, Class<E> clazz) { return i -> { try { consumer.accept(i); } catch (Exception ex) { try { E exCast = clazz.cast(ex); System.err.println( "Exception occured : " + exCast.getMessage()); } catch (ClassCastException ccEx) { throw ex; } } }; }
上面的類(lèi)傳入一個(gè)class,并將其cast到異常,如果能cast,則處理,否則拋出異常。
這樣處理之后,我們這樣調(diào)用:
ntegers.forEach( consumerWrapperWithExceptionClass( i -> System.out.println(1 / i), ArithmeticException.class));
處理checked Exception
checked Exception是必須要處理的異常,我們還是看個(gè)例子:
static void throwIOException(Integer integer) throws IOException { } List<Integer> integers = Arrays.asList(1, 2, 3, 4, 5); integers.forEach(i -> throwIOException(i));
上面我們定義了一個(gè)方法拋出IOException,這是一個(gè)checked Exception,需要被處理,所以在下面的forEach中,程序會(huì)編譯失敗,因?yàn)闆](méi)有處理相應(yīng)的異常。
最簡(jiǎn)單的辦法就是try,catch住,如下所示:
ntegers.forEach(i -> { try { throwIOException(i); } catch (IOException e) { throw new RuntimeException(e); } });
當(dāng)然,這樣的做法的壞處我們?cè)谏厦嬉呀?jīng)講過(guò)了,同樣的,我們可以定義一個(gè)新的wrapper方法:
static <T> Consumer<T> consumerWrapper( ThrowingConsumer<T, Exception> throwingConsumer) { return i -> { try { throwingConsumer.accept(i); } catch (Exception ex) { throw new RuntimeException(ex); } }; }
我們這樣調(diào)用:
integers.forEach(consumerWrapper(i -> throwIOException(i)));
我們也可以封裝一下異常:
static <T, E extends Exception> Consumer<T> consumerWrapperWithExceptionClass( ThrowingConsumer<T, E> throwingConsumer, Class<E> exceptionClass) { return i -> { try { throwingConsumer.accept(i); } catch (Exception ex) { try { E exCast = exceptionClass.cast(ex); System.err.println( "Exception occured : " + exCast.getMessage()); } catch (ClassCastException ccEx) { throw new RuntimeException(ex); } } }; }
然后這樣調(diào)用:
integers.forEach(consumerWrapperWithExceptionClass( i -> throwIOException(i), IOException.class));
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
- Java 進(jìn)階使用 Lambda 表達(dá)式實(shí)現(xiàn)超強(qiáng)的排序功能
- 深入理解Java8新特性之Lambda表達(dá)式的基本語(yǔ)法和自定義函數(shù)式接口
- Java中Lambda表達(dá)式的進(jìn)化之路詳解
- java Lambda表達(dá)式的使用心得
- Java中使用Lambda表達(dá)式和函數(shù)編程示例
- 一文帶你掌握J(rèn)ava8中Lambda表達(dá)式 函數(shù)式接口及方法構(gòu)造器數(shù)組的引用
- Java中Lambda表達(dá)式的使用詳解
- 吊打Java面試官之Lambda表達(dá)式 Stream API
- 深入淺出理解Java Lambda表達(dá)式之四大核心函數(shù)式的用法與范例
相關(guān)文章
JDK17在Windows安裝及環(huán)境變量配置超詳細(xì)的教程
這篇文章主要介紹了JDK17在Windows安裝及環(huán)境變量配置超詳細(xì)的教程,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-11-11SpringBoot整合Redis之編寫(xiě)RedisConfig
RedisConfig需要對(duì)redis提供的兩個(gè)Template的序列化配置,所以本文為大家詳細(xì)介紹了SpringBoot整合Redis如何編寫(xiě)RedisConfig,需要的可以參考下2022-06-06Java HashMap源碼及并發(fā)環(huán)境常見(jiàn)問(wèn)題解決
這篇文章主要介紹了Java HashMap源碼及并發(fā)環(huán)境常見(jiàn)問(wèn)題解決,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-09-09SpringBoot實(shí)現(xiàn)發(fā)送郵件功能
這篇文章主要介紹了SpringBoot 發(fā)送郵件功能實(shí)現(xiàn),本文以163郵箱為例通過(guò)這個(gè)小案例給大家介紹,需要的朋友可以參考下2019-12-12Spring?Boot集成Milvus快速入門(mén)demo示例詳解
Milvus是一種高性能向量數(shù)據(jù)庫(kù),支持從筆記本到大型分布式系統(tǒng)的多環(huán)境運(yùn)行,它以開(kāi)源和云服務(wù)形式提供,是LFAI & Data Foundation的項(xiàng)目,采用Apache 2.0許可,Milvus特別支持高并行化和解耦的系統(tǒng)架構(gòu),使其能夠隨數(shù)據(jù)增長(zhǎng)而擴(kuò)展,支持各種復(fù)雜搜索功能,滿足企業(yè)級(jí)AI應(yīng)用需求2024-09-09Java獲得一個(gè)數(shù)組的指定長(zhǎng)度排列組合算法示例
這篇文章主要介紹了Java獲得一個(gè)數(shù)組的指定長(zhǎng)度排列組合算法,結(jié)合實(shí)例形式分析了java排列組合相關(guān)數(shù)組遍歷、運(yùn)算操作技巧,需要的朋友可以參考下2019-06-06通過(guò)weblogic API解析如何獲取weblogic中服務(wù)的IP和端口操作
這篇文章主要介紹了通過(guò)weblogic API解析如何獲取weblogic中服務(wù)的IP和端口操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-06-06Java編程實(shí)現(xiàn)統(tǒng)計(jì)數(shù)組中各元素出現(xiàn)次數(shù)的方法
這篇文章主要介紹了Java編程實(shí)現(xiàn)統(tǒng)計(jì)數(shù)組中各元素出現(xiàn)次數(shù)的方法,涉及java針對(duì)數(shù)組的遍歷、比較、運(yùn)算等相關(guān)操作技巧,需要的朋友可以參考下2017-07-07java WebSocket客戶(hù)端斷線重連的實(shí)現(xiàn)方法
在工作中是否會(huì)遇到實(shí)用websocket客戶(hù)端連接服務(wù)端的時(shí)候,網(wǎng)絡(luò)波動(dòng),服務(wù)端斷連的情況,本文可以直接使用的斷線重連,感興趣的可以了解一下2021-10-10