在編程世界中,「空指針異常(NullPointerException)」無(wú)疑是我們最常遇到的"罪魁禍?zhǔn)?quot;之一。它像一片隱蔽的地雷,靜靜地等待著我們不小心地踏入,給我們的代碼帶來(lái)潛在的威脅。這種問(wèn)題雖然看似微小,但卻無(wú)法忽視。甚至可能對(duì)整個(gè)程序的穩(wěn)定性產(chǎn)生重大影響。
為了應(yīng)對(duì)這個(gè)長(zhǎng)久以來(lái)困擾開(kāi)發(fā)者的問(wèn)題,JAVA 8 版本引入了一個(gè)強(qiáng)大的工具——Optional 類。
Optional 不僅僅是一個(gè)容器,它更是一種編程理念的轉(zhuǎn)變,讓我們可以用更優(yōu)雅的方式處理可能為空的情況。
在本篇博客中,我將向大家介紹 JDK Optional 類及其使用方法,幫助你從根本上杜絕空指針異常,提升代碼質(zhì)量。
Optional 介紹
Optional 類是一個(gè)容器對(duì)象,它可以包含或不包含非空值。如果一個(gè)對(duì)象可能為空,那么我們就可以使用 Optional 類來(lái)代替該對(duì)象。
Optional 類型的變量可以有兩種狀態(tài):存在值和不存在值。
Optional類有兩個(gè)重要的方法:of和ofNullable:
-
of方法用于創(chuàng)建一個(gè)非空的Optional對(duì)象,如果傳入的參數(shù)為null,則會(huì)拋出NullPointerException異常。 -
ofNullable方法用于創(chuàng)建一個(gè)可以為空的Optional對(duì)象。如果傳入的參數(shù)為空,則返回一個(gè)空的Optional對(duì)象。當(dāng) Optional 對(duì)象存在值時(shí),調(diào)用 get() 方法可以返回該值,當(dāng) Optional 對(duì)象不存在值時(shí),調(diào)用 get() 方法會(huì)拋出 NoSuchElementException 異常。
下面是一個(gè)使用 Optional 類的例子:
Optional<String> optional = Optional.of("Hello World");
System.out.println(optional.get()); //輸出 Hello World
在上面的例子中,我們首先使用 of() 方法創(chuàng)建了一個(gè)包含字符串 "Hello World" 的 Optional 對(duì)象,然后使用 get() 方法獲取該對(duì)象的值并將其打印出來(lái)。
注意,如果我們嘗試創(chuàng)建一個(gè) null 值的 Optional 對(duì)象,則會(huì)拋出 NullPointerException 異常。
在使用 Optional 類時(shí),我們應(yīng)該盡量避免使用 isPresent()
和 get()
方法,因?yàn)檫@些方法可能會(huì)引起空指針異常。比較推薦使用Optional.ofNullable()來(lái)創(chuàng)建一個(gè)Optional 對(duì)象。
Optional 使用
創(chuàng)建 Optional 對(duì)象
我們可以使用以下幾種方式來(lái)創(chuàng)建 Optional 對(duì)象:
-
Optional.of(value)
:創(chuàng)建一個(gè)包含非空值的 Optional 對(duì)象。 -
Optional.empty()
:創(chuàng)建一個(gè)不包含任何值的空 Optional 對(duì)象。 -
Optional.ofNullable(value)
:創(chuàng)建一個(gè)可能包含 null 值的 Optional 對(duì)象。如果 value 不為 null,則該方法會(huì)創(chuàng)建一個(gè)包含該值的 Optional 對(duì)象;否則,創(chuàng)建一個(gè)空 Optional 對(duì)象。
下面是一個(gè)使用 Optional.ofNullable()
方法的例子:
String str = null;
Optional<String> optional = Optional.ofNullable(str);
System.out.println(optional.isPresent()); //輸出 false
在上面的例子中,我們首先聲明了一個(gè)空字符串 str,并將其賦值為 null。然后,我們使用 ofNullable()
方法創(chuàng)建了一個(gè) Optional 對(duì)象。由于 str 是 null,因此創(chuàng)建的 Optional 對(duì)象也是空的。最后,我們使用 isPresent()
方法檢查 Optional 對(duì)象是否包含值。
orElse()與orElseGet()
orElse()
方法接收一個(gè)參數(shù),即為默認(rèn)值。如果Optional對(duì)象中的值不為空,則返回該值,否則返回傳入的默認(rèn)值。具體用法如下:
Optional<String> optional = Optional.ofNullable(null);
String result = optional.orElse("default");
System.out.println(result); // 輸出 "default"
orElseGet()
方法與orElse()
方法類似,也是用于獲取默認(rèn)值的方法。但是,orElseGet()
方法接收的參數(shù)是一個(gè)「Supplier函數(shù)式接口」,用于在需要返回默認(rèn)值時(shí)生成該值。具體用法如下:
Optional<String> optional = Optional.ofNullable(null);
String result = optional.orElseGet(() -> "default");
System.out.println(result); // 輸出 "default"
orElse() 和 orElseGet()的區(qū)別
orElse()
和 orElseGet()
特別相似,有必要抽離出來(lái)講下他們之間的區(qū)別。
orElse()
方法無(wú)論 Optional 對(duì)象是否為空都會(huì)執(zhí)行,因此它總是會(huì)創(chuàng)建一個(gè)新的對(duì)象。orElseGet()
方法只有在 Optional 對(duì)象為空時(shí)才會(huì)執(zhí)行,因此它可以用來(lái)延遲創(chuàng)建新的對(duì)象。
用一個(gè)例子感受一下:
@Test
void test() {
System.out.println("--------不為null的情況----------");
//不為 null
String str1 = "hello";
String result11 = Optional.ofNullable(str1).orElse(get(str1 + ":orElse()方法被執(zhí)行了"));
String result12 = Optional.ofNullable(str1).orElseGet(() -> get(str1 + ":orElseGet()方法被執(zhí)行了"));
System.out.println(result11);
System.out.println(result12);
System.out.println("--------為null的情況----------");
//為 null
String str2 = null;
String result21 = Optional.ofNullable(str2).orElse(get(str1 + ":orElse()方法被執(zhí)行了"));
String result22 = Optional.ofNullable(str2).orElseGet(() -> get(str2 + ":orElseGet()方法被執(zhí)行了"));
System.out.println(result21);
System.out.println(result22);
}
public String get(String name) {
System.out.println(name);
return name;
}
輸出結(jié)果如下:
--------不為null的情況----------
hello:orElse()方法被執(zhí)行了
hello
hello
--------為null的情況----------
hello:orElse()方法被執(zhí)行了
null:orElseGet()方法被執(zhí)行了
hello:orElse()方法被執(zhí)行了
null:orElseGet()方法被執(zhí)行了
因此,一般來(lái)說(shuō),如果你希望在 Optional 對(duì)象為空時(shí)才創(chuàng)建新的對(duì)象,可以使用 orElseGet()
方法。否則,如果你希望總是創(chuàng)建新的對(duì)象,無(wú)論 Optional 對(duì)象是否為空,可以使用 orElse()
方法,通常來(lái)說(shuō)orElseGet()
更佳,個(gè)人也是推薦使用orElseGet()
。
map()與flatMap()
map()
方法接受一個(gè)函數(shù)作為參數(shù),該函數(shù)將被應(yīng)用于 Optional 對(duì)象中的值。如果 Optional 對(duì)象存在值,則將該值傳遞給函數(shù)進(jìn)行轉(zhuǎn)換,否則返回一個(gè)空 Optional 對(duì)象。
下面是一個(gè)使用 map() 方法的例子:
String str = "Hello";
Optional<String> optional = Optional.of(str);
Optional<String> upperCaseoptional = optional.map(String::toUpperCase);
System.out.println(upperCaseOptional.get()); //輸出 HELLO
在上面的例子中,我們首先使用 of()
方法創(chuàng)建了一個(gè)包含字符串 "Hello" 的 Optional 對(duì)象。然后,我們使用 map()
方法將該字符串轉(zhuǎn)換為大寫形式,并將結(jié)果存儲(chǔ)到另一個(gè) Optional 對(duì)象 upperCaseOptional 中。最后,我們使用 get()
方法獲取 upperCaseOptional 對(duì)象中的值并打印出來(lái)。
flatMap()
方法與 map()
方法類似,都接受一個(gè)函數(shù)作為參數(shù)。但是,flatMap()
方法返回的是一個(gè) Optional 類型的值。如果函數(shù)返回的是一個(gè) Optional 對(duì)象,則該方法會(huì)將其"展開(kāi)",否則返回一個(gè)空 Optional 對(duì)象。
下面是一個(gè)使用 flatMap() 方法的例子:
String str = "hello world";
Optional<String> optional = Optional.of(str);
Optional<Character> result = optional.flatMap(s -> {
if (s.length() > 0)
return Optional.of(s.charAt(0));
else
return Optional.empty();
});
System.out.println(result.get()); //輸出 h
在上面的例子中,我們首先創(chuàng)建了一個(gè)包含字符串 "hello world" 的 Optional 對(duì)象。然后,我們使用 flatMap()
方法將該字符串轉(zhuǎn)換為第一個(gè)字符,并將結(jié)果存儲(chǔ)到另一個(gè) Optional 對(duì)象 result 中。最后,我們使用 get()
方法獲取 result 對(duì)象中的值并打印出來(lái)。
filter()
filter() 方法接受一個(gè)「謂詞(Predicate)」作為參數(shù),返回一個(gè) Optional 類型的值。如果 Optional 對(duì)象存在值且滿足謂詞的條件,則返回該 Optional 對(duì)象,否則返回一個(gè)空 Optional 對(duì)象。
下面是一個(gè)使用 filter() 方法的例子:
Integer number = 10;
Optional<Integer> optional = Optional.of(number);
Optional<Integer> result = optional.filter(n -> n % 2 == 0);
System.out.println(result.isPresent()); //輸出 true
在上面的例子中,我們首先創(chuàng)建了一個(gè)包含整數(shù) 10 的 Optional 對(duì)象。然后,我們使用 filter()
方法過(guò)濾出該數(shù)字是否為偶數(shù),并將結(jié)果存儲(chǔ)到另一個(gè) Optional 對(duì)象 result 中。最后,我們使用 isPresent()
方法檢查 result 對(duì)象是否存在值。
常用方法
我們已經(jīng)介紹了Optional類的幾種常用方法。除此之外,我們這里再逐一列舉和解析其他方法。
方法名 | 說(shuō)明 |
---|---|
empty() |
返回一個(gè)空的Optional 實(shí)例。 |
of(T value) |
返回一個(gè)包含給定非null值的Optional 。如果value為null,會(huì)拋出NullPointerException。 |
ofNullable(T value) |
返回一個(gè)可選的包含給定值的Optional ,也可能為空。 |
get() |
如果Optional 有值則將其返回,否則拋出NoSuchElementException 。 |
isPresent() |
如果值存在返回true,否則false。 |
isEmpty() |
如果值不存在返回true,否則false。(Java 11后新增) |
ifPresent(Consumer<? super T> consumer) |
如果值存在,則使用該值調(diào)用指定的消費(fèi)者,否則不執(zhí)行任何操作。 |
ifPresentOrElse(Consumer<? super T> action, Runnable emptyAction) |
如果值存在,則使用該值調(diào)用指定的消費(fèi)者,否則執(zhí)行指定的無(wú)參數(shù)的動(dòng)作。(Java 9后新增) |
filter(Predicate<? super T> predicate) |
如果一個(gè)值存在,并且這個(gè)值給定的predicate對(duì)其返回true,返回一個(gè)Optional 描述的值,否則返回一個(gè)空的Optional 。 |
map(Function<? super T,? extends U> mApper) |
如果有值,應(yīng)用mapping函數(shù)并返回結(jié)果。否則返回空的Optional 。 |
flatMap(Function<? super T,Optional<U>> mapper) |
如果值存在,返回從Optional中提取的值,否則返回一個(gè)空的Optional 。 |
orElse(T other) |
如果存在該值,返回值, 否則返回other 。 |
orElseGet(Supplier<? extends T> other) |
如果存在該值,返回值,否則觸發(fā)other 并返回。 |
orElseThrow() |
如果存在該值,返回包含的值,否則拋出NoSuchElementException 。 |
orElseThrow(Supplier<? extends X> exceptionSupplier) |
如果存在該值,返回包含的值,否則拋出由Supplier 產(chǎn)生的異常。 |
在這篇文章中,我們深入探討了Java的Optional類及其在編程實(shí)踐中的應(yīng)用。通過(guò)使用Optional,我們可以更有效地處理可能存在的空值情況,從而避免運(yùn)行時(shí)的NullPointException。雖然它引入了額外的復(fù)雜性,但如果正確使用,它可以提供更清晰、更易于維護(hù)的代碼。
但是,重要的是要記住,Optional并不是解決所有問(wèn)題的銀彈。像所有工具一樣,我們需要了解它的優(yōu)點(diǎn)和局限性,并確保在適當(dāng)?shù)膱?chǎng)景下使用它。
編程始終是一個(gè)學(xué)習(xí)和探索的過(guò)程,Optional只是我們工具箱中的一個(gè)工具。希望通過(guò)本文,你對(duì)如何利用Java的Optional類有了更全面的理解。