java 8 optional

這篇文章寫得不錯(cuò),所以轉(zhuǎn)載了下,修改了小部分,原文地址見末尾

身為一名Java程序員,大家可能都有這樣的經(jīng)歷:調(diào)用一個(gè)方法得到了返回值卻不能直接將返回值作為參數(shù)去調(diào)用別的方法。我們首先要判斷這個(gè)返回值是否為null,只有在非空的前提下才能將其作為其他方法的參數(shù)。這正是一些類似Guava的外部API試圖解決的問題。一些JVM編程語言比如Scala、Ceylon等已經(jīng)將對在核心API中解決了這個(gè)問題。
新版本的Java,比如Java 8引入了一個(gè)新的Optional類。Optional類的Javadoc描述如下:
這是一個(gè)可以為null的容器對象。如果值存在則isPresent()方法會(huì)返回true,調(diào)用get()方法會(huì)返回該對象。

本文會(huì)逐個(gè)探討Optional類包含的方法,并通過一兩個(gè)示例展示如何使用。

of

為非null的值創(chuàng)建一個(gè)Optional。of方法通過工廠方法創(chuàng)建Optional類。需要注意的是,創(chuàng)建對象時(shí)傳入的參數(shù)不能為null。如果傳入?yún)?shù)為null,則拋出NullPointerException 。

//調(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的情況。示例如下:

//下面創(chuàng)建了一個(gè)不包含任何值的Optional實(shí)例
//例如,值為'null'
Optional empty = Optional.ofNullable(null);

isPresent

非常容易理解,如果值存在返回true,否則返回false。類似下面的代碼:

//isPresent方法用來檢查Optional實(shí)例中是否包含值
if(name.isPresent()) {
  //在Optional實(shí)例內(nèi)調(diào)用get()返回已存在的值
  System.out.println(name.get());
  //輸出Sanaulla
}

get

如果Optional有值則將其返回,否則拋出NoSuchElementException。

上面的示例中,get方法用來得到Optional實(shí)例中的值。下面我們看一個(gè)拋出NoSuchElementException的例子:

//執(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類。簡答地說,Consumer類包含一個(gè)抽象方法。該抽象方法對傳入的值進(jìn)行處理,但沒有返回值。Java8支持不用接口直接通過lambda表達(dá)式傳入?yún)?shù)。
如果Optional實(shí)例有值,調(diào)用ifPresent()可以接受接口段或lambda表達(dá)式。類似下面的代碼:

//ifPresent方法接受lambda表達(dá)式作為參數(shù)。
//lambda表達(dá)式對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ù)。示例如下:

//如果值不為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)用來生成默認(rèn)值。示例如下:

//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á)式或方法,如果值不存在來拋出異常。示例如下:

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定義如下:

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方法文檔說明如下:
如果有值,則對其執(zhí)行調(diào)用mapping函數(shù)得到返回值。如果返回值不為null,則創(chuàng)建包含mapping返回值的Optional作為map方法返回值,否則返回空Optional。

map方法用來對Optional實(shí)例的值執(zhí)行一系列操作。通過一組實(shí)現(xiàn)了Function接口的lambda表達(dá)式傳入操作。如果你不熟悉Function接口,可以參考我的這篇博客。map方法示例如下:

//map方法執(zhí)行傳入的lambda表達(dá)式參數(shù)對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ì)對結(jié)果用Optional封裝。

flatMap方法與map方法類似,區(qū)別在于mapping函數(shù)的返回值不同。map方法的mapping函數(shù)返回值可以是任何類型T,而flatMap方法的mapping函數(shù)必須是Optional。
參照map函數(shù),使用flatMap重寫的示例如下:

//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è)方法通過傳入限定條件對Optional實(shí)例的值進(jìn)行過濾。文檔描述如下:
如果有值并且滿足斷言條件返回包含該值的Optional,否則返回空Optional。

讀到這里,可能你已經(jīng)知道如何為filter方法傳入一段代碼。是的,這里可以傳入一個(gè)lambda表達(dá)式。對于filter函數(shù)我們應(yīng)該傳入實(shí)現(xiàn)了Predicate接口的lambda表達(dá)式。如果你不熟悉Predicate接口,可以參考這篇文章。

現(xiàn)在我來看看filter的各種用法,下面的示例介紹了滿足限定條件和不滿足兩種情況:

//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長度不足6字符

System.out.println(shortName.orElse(
"The name is less than 6 characters"
));

以上,我們介紹了Optional類的各個(gè)方法。下面通過一個(gè)完整的示例對用法集中展示:


public class OptionalDemo {  
public static void main(String[] args) {
  //創(chuàng)建Optional實(shí)例,也可以通過方法返回值得到。    
  Optional<String> name = Optional.of(
  "Sanaulla"
);
  
//創(chuàng)建沒有值的Optional實(shí)例,例如值為'null'
    
Optional empty = Optional.ofNullable(null); 
//isPresent方法用來檢查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方法通過傳入的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"));   
}
}

上述代碼輸出如下:

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

原文鏈接: javacodegeeks 翻譯: ImportNew.com - 高俊陽譯文鏈接: http://www.importnew.com/6675.html

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容