選取單個(gè)元素
直覺來說選取單個(gè)元素肯定會(huì)比選取多個(gè)要簡單得多,不過這里也存在一些問題。我們先看下一般的做法的問題是什么,然后再看下如何用lambda表達(dá)式來解決它。
我們先新建一個(gè)方法來查找一個(gè)以特定字母開頭的元素,然后打印出來。
public static void pickName(
final List<String> names, final String startingLetter) {
String foundName = null;
for(String name : names) {
if(name.startsWith(startingLetter)) {
foundName = name;
break;
}
}
這個(gè)方法簡直跟剛過去的垃圾車一樣臭不可聞。我們先是新建了一個(gè)foundName的變量,然后初始化成null——這個(gè)就是惡臭之源。我們不得不檢查是否為空,不然的話就會(huì)拋出一個(gè)NullPointerException或者一個(gè)錯(cuò)誤響應(yīng)。我們還用了一個(gè)外部迭代器來循環(huán)列表,如果找到了想要的元素之后還得跳出這個(gè)循環(huán),這又加重了原來的臭味:基本類型偏執(zhí),命令式風(fēng)格,可變性,全齊活了。一旦退出循環(huán)后,我們還得先檢查下結(jié)果,然后才能進(jìn)行打印。這么點(diǎn)任務(wù)居然寫了這么長的代碼。
我們來重新分析下這個(gè)問題。我們只是希望能選出第一個(gè)匹配的元素,并且能安全的處理不存在這樣一個(gè)元素的情況。我們來用lambda表達(dá)式重寫一下這個(gè)pickName方法。
public static void pickName(
final List<String> names, final String startingLetter) {
final Optional<String> foundName =
names.stream()
.filter(name ->name.startsWith(startingLetter))
.findFirst();
System.out.println(String.format("A name starting with %s: %s",
startingLetter, foundName.orElse("No name found")));
}
JDK里面一些強(qiáng)大的功能使得這段代碼更得非常簡潔。首先我們用filter方法獲取了所有滿足條件的元素,然后用了Stream類的findFirst方法選取出了返回集合的第一個(gè)元素。這個(gè)方法返回的是一個(gè)Optional對象,這就是Java里面官方認(rèn)證的null變量的除臭劑。
Optional類非常有用,你不用管結(jié)果是不是存在。它使得我們免受空指針異常的煩惱,并且更明確的指明了沒有結(jié)果也是一種可能的結(jié)果。通過isPresent()方法我們可以知道結(jié)果是不是存在,想獲取結(jié)果值的話可以使用get()方法。我們還可以使用(這個(gè)方法名能讓你震驚)orElse方法給它設(shè)置一個(gè)默認(rèn)值,就像前面代碼里的那樣。
我們用之前一直在用的friends集合來驗(yàn)證下我們這個(gè)pickName方法。
pickName(friends, "N");
pickName(friends, "Z");
這段代碼選取出第一個(gè)匹配的元素,如果沒找到,打印出一個(gè)友好的提示信息。
A name starting with N: Nate
A name starting with Z: No name found
findFirst()方法和Optinal類的結(jié)合使用減少了我們的代碼量,并且看起來感覺還不錯(cuò)。不過Optional類的功能遠(yuǎn)不止這些。比如說,除了當(dāng)對象不存在的時(shí)候能提供一個(gè)默認(rèn)值外,如果結(jié)果存在的話還可以用它來運(yùn)行一段代碼,或者一個(gè)lambda表達(dá)式,像這樣:
foundName.ifPresent(name -> System.out.println("Hello " + name));
跟命令式的選取第一個(gè)匹配名字的代碼比起來,流式的優(yōu)雅的函數(shù)式風(fēng)格看真來更棒一些。不過這個(gè)調(diào)用流的版本里是不是做的事情有點(diǎn)太多了(譯注:先選出了所有匹配的再返回第一項(xiàng))?當(dāng)然不是,這些方法非常智能,它們可以按需工作(在后面113頁的Stream的惰性求值中我們會(huì)深入探討這點(diǎn))。
選取單個(gè)元素的例子展示了JDK庫更多強(qiáng)大的功能,下面我們來看下lambda表達(dá)式如何根據(jù)一個(gè)集合,來求出一個(gè)想要的值。