前言
身為一名Java程序員,大家可能都有這樣的經歷:調用一個方法得到了返回值卻不能直接將返回值作為參數去調用別的方法。我們首先要判斷這個返回值是否為null,只有在非空的前提下才能將其作為其他方法的參數。這正是一些類似Guava的外部API試圖解決的問題。一些JVM編程語言比如Scala、Ceylon等已經將對在核心API中解決了這個問題。
Optional概述
新版本的Java,比如Java 8引入了一個新的Optional類。Optional類的Javadoc描述如下:
這是一個可以為null的容器對象。如果值存在則isPresent()方法會返回true,調用get()方法會返回該對象。
本文會逐個探討Optional類包含的方法,並通過一兩個示例展示如何使用。
實例演示
of
為非null的值創建一個Optional。
of方法通過工廠方法創建Optional類。需要注意的是,創建對象時傳入的參數不能為null。如果傳入參數為null,則拋出NullPointerException 。
ofNullable
為指定的值創建一個Optional,如果指定的值為null,則返回一個空的Optional。
isPresent
如果值存在返回true,否則返回false。
get
如果Optional有值則將其返回,否則拋出NoSuchElementException。
ifPresent
如果Optional實例有值則為其調用consumer,否則不做處理
orElse
如果有值則將其返回,否則返回指定的其它值。
orElseGet
orElseGet與orElse方法類似,區別在於得到的默認值。orElse方法將傳入的字符串作為默認值,orElseGet方法可以接受Supplier接口的實現用來生成默認值
orElseThrow
如果有值則將其返回,否則拋出supplier接口創建的異常。
在orElseGet方法中,我們傳入一個Supplier接口。然而,在orElseThrow中我們可以傳入一個lambda表達式或方法,如果值不存在來拋出異常。
map
如果有值,則對其執行調用mapping函數得到返回值。如果返回值不為null,則創建包含mapping返回值的Optional作為map方法返回值,否則返回空Optional。
map方法用來對Optional實例的值執行一系列操作。通過一組實現了Function接口的lambda表達式傳入操作。
flatMap
如果有值,為其執行mapping函數返回Optional類型返回值,否則返回空Optional。
flatMap方法與map方法類似,區別在於mapping函數的返回值不同。map方法的mapping函數返回值可以是任何類型T,而flatMap方法的mapping函數必須是Optional。
參照map函數,使用flatMap重寫的示例如下:
filter
如果有值並且滿足斷言條件返回包含該值的Optional,否則返回空Optional。
這里可以傳入一個lambda表達式。對於filter函數我們應該傳入實現了Predicate接口的lambda表達式。
完整示例
/* of */
//調用工廠方法創建Optional實例
Optional<String> name = Optional.of("YanWei");
//傳入參數為null,拋出NullPointerException.
//Optional<String> someNull = Optional.of(null);
/*opNullable*/
Optional empty = Optional.ofNullable(null);
/*isPresent*/
if (name.isPresent()) {
System.out.println(name.get());//輸出YanWei
}
/*get*/
try {
System.out.println(empty.get());
} catch (NoSuchElementException ex) {
System.err.println(ex.getMessage());
}
/*ifPresent*/
name.ifPresent((value) -> {
System.out.println("The length of the value is: " + value.length());
});
/*orElse*/
System.out.println(empty.orElse("There is no value present!"));
System.out.println(name.orElse("There is some value!"));
/*orElseGet*/
System.out.println(empty.orElseGet(() -> "Default Value"));
System.out.println(name.orElseGet(String::new));
/*orElseThrow*/
try {
empty.orElseThrow(IllegalArgumentException::new);
} catch (Throwable ex) {
System.out.println("error:" + ex.getMessage());
}
/*map*/
Optional<String> upperName = name.map((value) -> value.toUpperCase());
System.out.println(upperName.orElse("No value found"));
/*flatMap*/
upperName = name.flatMap((value) -> Optional.of(value.toUpperCase()));
System.out.println(upperName.get());
/*filter*/
List<String> names = Arrays.asList("YanWei","YanTian");
for(String s:names)
{
Optional<String> nameLenLessThan7 = Optional.of(s).filter((value) -> value.length() < 7);
System.out.println(nameLenLessThan7.orElse("The name is more than 6 characters"));
}