Java 8 Optional类深度解析
轉(zhuǎn)載自?http://www.importnew.com/6675.html
身為一名Java程序員,大家可能都有這樣的經(jīng)歷:調(diào)用一個(gè)方法得到了返回值卻不能直接將返回值作為參數(shù)去調(diào)用別的方法。我們首先要判斷這個(gè)返回值是否為null,只有在非空的前提下才能將其作為其他方法的參數(shù)。這正是一些類似Guava的外部API試圖解決的問(wèn)題。一些JVM編程語(yǔ)言比如Scala、Ceylon等已經(jīng)將對(duì)在核心API中解決了這個(gè)問(wèn)題。在我的前一篇文章中,介紹了Scala是如何解決了這個(gè)問(wèn)題。
新版本的Java,比如Java 8引入了一個(gè)新的Optional類。Optional類的Javadoc描述如下:
這是一個(gè)可以為null的容器對(duì)象。如果值存在則isPresent()方法會(huì)返回true,調(diào)用get()方法會(huì)返回該對(duì)象。
本文會(huì)逐個(gè)探討Optional類包含的方法,并通過(guò)一兩個(gè)示例展示如何使用。
of
為非null的值創(chuàng)建一個(gè)Optional。
of方法通過(guò)工廠方法創(chuàng)建Optional類。需要注意的是,創(chuàng)建對(duì)象時(shí)傳入的參數(shù)不能為null。如果傳入?yún)?shù)為null,則拋出NullPointerException 。
| 1 2 3 4 | //調(diào)用工廠方法創(chuàng)建Optional實(shí)例 Optional<String> name = Optional.of("Sanaulla"); //傳入?yún)?shù)為null,拋出NullPointerException. Optional<String> someNull = Optional.of(null); |
ofNullable
為指定的值創(chuàng)建一個(gè)Optional,如果指定的值為null,則返回一個(gè)空的Optional。
ofNullable與of方法相似,唯一的區(qū)別是可以接受參數(shù)為null的情況。示例如下:
| 1 2 3 | //下面創(chuàng)建了一個(gè)不包含任何值的Optional實(shí)例 //例如,值為'null' Optional empty = Optional.ofNullable(null); |
isPresent
非常容易理解
如果值存在返回true,否則返回false。
類似下面的代碼:
| 1 2 3 4 5 | //isPresent方法用來(lái)檢查Optional實(shí)例中是否包含值 if?(name.isPresent()) { ??//在Optional實(shí)例內(nèi)調(diào)用get()返回已存在的值 ??System.out.println(name.get());//輸出Sanaulla } |
get
如果Optional有值則將其返回,否則拋出NoSuchElementException。
上面的示例中,get方法用來(lái)得到Optional實(shí)例中的值。下面我們看一個(gè)拋出NoSuchElementException的例子:
| 1 2 3 4 5 6 7 | //執(zhí)行下面的代碼會(huì)輸出:No value present try?{ ??//在空的Optional實(shí)例上調(diào)用get(),拋出NoSuchElementException ??System.out.println(empty.get()); }?catch?(NoSuchElementException ex) { ??System.out.println(ex.getMessage()); } |
ifPresent
如果Optional實(shí)例有值則為其調(diào)用consumer,否則不做處理
要理解ifPresent方法,首先需要了解Consumer類。簡(jiǎn)答地說(shuō),Consumer類包含一個(gè)抽象方法。該抽象方法對(duì)傳入的值進(jìn)行處理,但沒(méi)有返回值。Java8支持不用接口直接通過(guò)lambda表達(dá)式傳入?yún)?shù)。
如果Optional實(shí)例有值,調(diào)用ifPresent()可以接受接口段或lambda表達(dá)式。類似下面的代碼:
| 1 2 3 4 5 | //ifPresent方法接受lambda表達(dá)式作為參數(shù)。 //lambda表達(dá)式對(duì)Optional的值調(diào)用consumer進(jìn)行處理。 name.ifPresent((value) -> { ??System.out.println("The length of the value is: "?+ value.length()); }); |
orElse
如果有值則將其返回,否則返回指定的其它值。
如果Optional實(shí)例有值則將其返回,否則返回orElse方法傳入的參數(shù)。示例如下:
| 1 2 3 4 5 6 | //如果值不為null,orElse方法返回Optional實(shí)例的值。 //如果為null,返回傳入的消息。 //輸出:There is no value present! System.out.println(empty.orElse("There is no value present!")); //輸出:Sanaulla System.out.println(name.orElse("There is some value!")); |
orElseGet
orElseGet與orElse方法類似,區(qū)別在于得到的默認(rèn)值。orElse方法將傳入的字符串作為默認(rèn)值,orElseGet方法可以接受Supplier接口的實(shí)現(xiàn)用來(lái)生成默認(rèn)值。示例如下:
| 1 2 3 4 5 6 | //orElseGet與orElse方法類似,區(qū)別在于orElse傳入的是默認(rèn)值, //orElseGet可以接受一個(gè)lambda表達(dá)式生成默認(rèn)值。 //輸出:Default Value System.out.println(empty.orElseGet(() ->?"Default Value")); //輸出:Sanaulla System.out.println(name.orElseGet(() ->?"Default Value")); |
orElseThrow
如果有值則將其返回,否則拋出supplier接口創(chuàng)建的異常。
在orElseGet方法中,我們傳入一個(gè)Supplier接口。然而,在orElseThrow中我們可以傳入一個(gè)lambda表達(dá)式或方法,如果值不存在來(lái)拋出異常。示例如下:
| 1 2 3 4 5 6 7 8 9 | try?{ ??//orElseThrow與orElse方法類似。與返回默認(rèn)值不同, ??//orElseThrow會(huì)拋出lambda表達(dá)式或方法生成的異常 ? ??empty.orElseThrow(ValueAbsentException::new); }?catch?(Throwable ex) { ??//輸出: No value present in the Optional instance ??System.out.println(ex.getMessage()); } |
ValueAbsentException定義如下:
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 | class?ValueAbsentException?extends?Throwable { ? ??public?ValueAbsentException() { ????super(); ??} ? ??public?ValueAbsentException(String msg) { ????super(msg); ??} ? ??@Override ??public?String getMessage() { ????return?"No value present in the Optional instance"; ??} } |
map
map方法文檔說(shuō)明如下:
如果有值,則對(duì)其執(zhí)行調(diào)用mapping函數(shù)得到返回值。如果返回值不為null,則創(chuàng)建包含mapping返回值的Optional作為map方法返回值,否則返回空Optional。
map方法用來(lái)對(duì)Optional實(shí)例的值執(zhí)行一系列操作。通過(guò)一組實(shí)現(xiàn)了Function接口的lambda表達(dá)式傳入操作。如果你不熟悉Function接口,可以參考我的這篇博客。map方法示例如下:
| 1 2 3 4 | //map方法執(zhí)行傳入的lambda表達(dá)式參數(shù)對(duì)Optional實(shí)例的值進(jìn)行修改。 //為lambda表達(dá)式的返回值創(chuàng)建新的Optional實(shí)例作為map方法的返回值。 Optional<String> upperName = name.map((value) -> value.toUpperCase()); System.out.println(upperName.orElse("No value found")); |
flatMap
如果有值,為其執(zhí)行mapping函數(shù)返回Optional類型返回值,否則返回空Optional。flatMap與map(Funtion)方法類似,區(qū)別在于flatMap中的mapper返回值必須是Optional。調(diào)用結(jié)束時(shí),flatMap不會(huì)對(duì)結(jié)果用Optional封裝。
flatMap方法與map方法類似,區(qū)別在于mapping函數(shù)的返回值不同。map方法的mapping函數(shù)返回值可以是任何類型T,而flatMap方法的mapping函數(shù)必須是Optional。
參照map函數(shù),使用flatMap重寫(xiě)的示例如下:
| 1 2 3 4 5 | //flatMap與map(Function)非常類似,區(qū)別在于傳入方法的lambda表達(dá)式的返回類型。 //map方法中的lambda表達(dá)式返回值可以是任意類型,在map函數(shù)返回之前會(huì)包裝為Optional。 //但flatMap方法中的lambda表達(dá)式返回值必須是Optionl實(shí)例。 upperName = name.flatMap((value) -> Optional.of(value.toUpperCase())); System.out.println(upperName.orElse("No value found"));//輸出SANAULLA |
filter
filter個(gè)方法通過(guò)傳入限定條件對(duì)Optional實(shí)例的值進(jìn)行過(guò)濾。文檔描述如下:
如果有值并且滿足斷言條件返回包含該值的Optional,否則返回空Optional。
讀到這里,可能你已經(jīng)知道如何為filter方法傳入一段代碼。是的,這里可以傳入一個(gè)lambda表達(dá)式。對(duì)于filter函數(shù)我們應(yīng)該傳入實(shí)現(xiàn)了Predicate接口的lambda表達(dá)式。如果你不熟悉Predicate接口,可以參考這篇文章。
現(xiàn)在我來(lái)看看filter的各種用法,下面的示例介紹了滿足限定條件和不滿足兩種情況:
| 1 2 3 4 5 6 7 8 9 10 | //filter方法檢查給定的Option值是否滿足某些條件。 //如果滿足則返回同一個(gè)Option實(shí)例,否則返回空Optional。 Optional<String> longName = name.filter((value) -> value.length() >?6); System.out.println(longName.orElse("The name is less than 6 characters"));//輸出Sanaulla ? //另一個(gè)例子是Optional值不滿足filter指定的條件。 Optional<String> anotherName = Optional.of("Sana"); Optional<String> shortName = anotherName.filter((value) -> value.length() >?6); //輸出:name長(zhǎng)度不足6字符 System.out.println(shortName.orElse("The name is less than 6 characters")); |
以上,我們介紹了Optional類的各個(gè)方法。下面通過(guò)一個(gè)完整的示例對(duì)用法集中展示:
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 | public?class?OptionalDemo { ? ??public?static?void?main(String[] args) { ????//創(chuàng)建Optional實(shí)例,也可以通過(guò)方法返回值得到。 ????Optional<String> name = Optional.of("Sanaulla"); ? ????//創(chuàng)建沒(méi)有值的Optional實(shí)例,例如值為'null' ????Optional empty = Optional.ofNullable(null); ? ????//isPresent方法用來(lái)檢查Optional實(shí)例是否有值。 ????if?(name.isPresent()) { ??????//調(diào)用get()返回Optional值。 ??????System.out.println(name.get()); ????} ? ????try?{ ??????//在Optional實(shí)例上調(diào)用get()拋出NoSuchElementException。 ??????System.out.println(empty.get()); ????}?catch?(NoSuchElementException ex) { ??????System.out.println(ex.getMessage()); ????} ? ????//ifPresent方法接受lambda表達(dá)式參數(shù)。 ????//如果Optional值不為空,lambda表達(dá)式會(huì)處理并在其上執(zhí)行操作。 ????name.ifPresent((value) -> { ??????System.out.println("The length of the value is: "?+ value.length()); ????}); ? ????//如果有值orElse方法會(huì)返回Optional實(shí)例,否則返回傳入的錯(cuò)誤信息。 ????System.out.println(empty.orElse("There is no value present!")); ????System.out.println(name.orElse("There is some value!")); ? ????//orElseGet與orElse類似,區(qū)別在于傳入的默認(rèn)值。 ????//orElseGet接受lambda表達(dá)式生成默認(rèn)值。 ????System.out.println(empty.orElseGet(() ->?"Default Value")); ????System.out.println(name.orElseGet(() ->?"Default Value")); ? ????try?{ ??????//orElseThrow與orElse方法類似,區(qū)別在于返回值。 ??????//orElseThrow拋出由傳入的lambda表達(dá)式/方法生成異常。 ??????empty.orElseThrow(ValueAbsentException::new); ????}?catch?(Throwable ex) { ??????System.out.println(ex.getMessage()); ????} ? ????//map方法通過(guò)傳入的lambda表達(dá)式修改Optonal實(shí)例默認(rèn)值。 ????//lambda表達(dá)式返回值會(huì)包裝為Optional實(shí)例。 ????Optional<String> upperName = name.map((value) -> value.toUpperCase()); ????System.out.println(upperName.orElse("No value found")); ? ????//flatMap與map(Funtion)非常相似,區(qū)別在于lambda表達(dá)式的返回值。 ????//map方法的lambda表達(dá)式返回值可以是任何類型,但是返回值會(huì)包裝成Optional實(shí)例。 ????//但是flatMap方法的lambda返回值總是Optional類型。 ????upperName = name.flatMap((value) -> Optional.of(value.toUpperCase())); ????System.out.println(upperName.orElse("No value found")); ? ????//filter方法檢查Optiona值是否滿足給定條件。 ????//如果滿足返回Optional實(shí)例值,否則返回空Optional。 ????Optional<String> longName = name.filter((value) -> value.length() >?6); ????System.out.println(longName.orElse("The name is less than 6 characters")); ? ????//另一個(gè)示例,Optional值不滿足給定條件。 ????Optional<String> anotherName = Optional.of("Sana"); ????Optional<String> shortName = anotherName.filter((value) -> value.length() >?6); ????System.out.println(shortName.orElse("The name is less than 6 characters")); ? ??} ? } |
上述代碼輸出如下:
| 1 2 3 4 5 6 7 8 9 10 11 12 | Sanaulla No value present The length of the value is: 8 There is no value present! Sanaulla Default Value Sanaulla No value present?in?the Optional instance SANAULLA SANAULLA Sanaulla The name is?less?than 6 characters |
總結(jié)
以上是生活随笔為你收集整理的Java 8 Optional类深度解析的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: linux 安装tomcat遇到的问题
- 下一篇: java美元兑换,(Java实现) 美元