Optional 類是一個可以為null的容器對象。如果值存在則isPresent()方法會返回true,調用get()方法會返回該對象。
Optional 是個容器:它可以保存類型T的值,或者僅僅保存null。Optional提供很多有用的方法,這樣我們就不用顯式進行空值檢測。
Optional 類的引入很好的解決空指針異常。
示例
public class Java8Tester {
public static void main(String args[]){
Java8Tester java8Tester = new Java8Tester();
Integer value1 = null;
Integer value2 = new Integer(10);
// Optional.ofNullable - 允許傳遞為 null 參數
Optional<Integer> a = Optional.ofNullable(value1);
// Optional.of - 如果傳遞的參數是 null,拋出異常 NullPointerException
Optional<Integer> b = Optional.of(value2);
System.out.println(java8Tester.sum(a,b));
}
public Integer sum(Optional<Integer> a, Optional<Integer> b){
// Optional.isPresent - 判斷值是否存在
System.out.println("第一個參數值存在: " + a.isPresent());
System.out.println("第二個參數值存在: " + b.isPresent());
// Optional.orElse - 如果值存在,返回它,否則返回默認值
Integer value1 = a.orElse(new Integer(0));
//Optional.get - 獲取值,值需要存在
Integer value2 = b.get();
return value1 + value2;
}
}
執行以上腳本,輸出結果為:
第一個參數值存在: false 第二個參數值存在: true 10
分析
1、上面例子僅說明了option類的用法,看起來好像並沒有什么用,以前拋NullPointerException,現在還是可能拋NullPointerException、NoSuchElementException等,感覺還不如以前的==null好使。接下來我們看看option類的方法。
類方法
| 序號 | 方法 & 描述 |
|---|---|
| 1 | static <T> Optional<T> empty() 返回空的 Optional 實例。 |
| 2 | boolean equals(Object obj) 判斷其他對象是否等於 Optional。 |
| 3 | Optional<T> filter(Predicate<? super <T> predicate) 如果值存在,並且這個值匹配給定的 predicate,返回一個Optional用以描述這個值,否則返回一個空的Optional。 |
| 4 | <U> Optional<U> flatMap(Function<? super T,Optional<U>> mapper) 如果值存在,返回基於Optional包含的映射方法的值,否則返回一個空的Optional |
| 5 | T get() 如果在這個Optional中包含這個值,返回值,否則拋出異常:NoSuchElementException |
| 6 | int hashCode() 返回存在值的哈希碼,如果值不存在 返回 0。 |
| 7 | void ifPresent(Consumer<? super T> consumer) 如果值存在則使用該值調用 consumer , 否則不做任何事情。 |
| 8 | boolean isPresent() 如果值存在則方法會返回true,否則返回 false。 |
| 9 | <U>Optional<U> map(Function<? super T,? extends U> mapper) 如果有值,則對其執行調用映射函數得到返回值。如果返回值不為 null,則創建包含映射返回值的Optional作為map方法返回值,否則返回空Optional。 |
| 10 | static <T> Optional<T> of(T value) 返回一個指定非null值的Optional。 |
| 11 | static <T> Optional<T> ofNullable(T value) 如果為非空,返回 Optional 描述的指定值,否則返回空的 Optional。 |
| 12 | T orElse(T other) 如果存在該值,返回值, 否則返回 other。 |
| 13 | T orElseGet(Supplier<? extends T> other) 如果存在該值,返回值, 否則觸發 other,並返回 other 調用的結果。 |
| 14 | <X extends Throwable> T orElseThrow(Supplier<? extends X> exceptionSupplier)
如果存在該值,返回包含的值,否則拋出由 Supplier 繼承的異常 |
| 15 | String toString() 返回一個Optional的非空字符串,用來調試 |
注意: 這些方法是從 java.lang.Object 類繼承來的。
上述的我標紅的方法可以看出一些端倪,發現都是些高階函數(高階函數是指接受另外一個函數作為參數,或返回一個函數的函數),而且很容易看出入參都是這次java8新加的函數式接口。所以這里應該就能推測出,哦,原來這個類基本上是給函數式編程用的。
下面舉個函數式編程用option和不用option的區別的🌰
@RequiredArgsConstructor
@AllArgsConstructor
@Getter
public class Consumer1 {
@Setter
private Order order;
}
@RequiredArgsConstructor
@AllArgsConstructor
@Getter
public class Order {
@Setter
private Product product;
}
@RequiredArgsConstructor
@AllArgsConstructor
@Getter
public class Product {
@Setter
private String name;
}
@Test
public void testOption() {
Product product = new Product("電腦");
Order order = new Order(product);
Consumer1 consumer = new Consumer1(order);
//java8寫法
String name = Optional.of(consumer)
.map(Consumer1::getOrder)
.map(Order::getProduct)
.map(Product::getName)
.orElse(null);
System.out.println("option 鏈式:"+name);
//java8之前寫法
name = null;
if (consumer != null) {
Order o = consumer.getOrder();
if (o != null) {
Product p = o.getProduct();
if (p != null) {
name = p.getName();
}
}
}
System.out.println("java8之前寫法:"+name);
}
這樣一看,Optional就顯得有優勢了,清楚明了。
2、既然Optional用起來還不錯,干脆定義為屬性類型好了,這個是不建議的,因為Optional沒有繼承Serializable。
不要將Optional作為方法參數進行定義,也不要在類中聲明Optional類型的變量,Optional通常只做為方法的返回值來規避空指針問題。

