一文教你搞定Java Optional類判空操作

概述

最近項目組內做code review,充斥著大量的、原始的、醜陋的判空語句,大致類似下面的代碼:

if (user != null) {
    Address address = user.getAddress();
    if (address != null) {
        Country country = address.getCountry();
        if (country != null) {
            String isocode = country.getIsocode();
            if (isocode != null) {
                isocode = isocode.toUpperCase();
            }
        }
    }
}

讓整體的代碼顯得十分的臃腫龐大醜陋,那麼怎麼辦呢?其實一行代碼就可以搞定:

String result = Optional.ofNullable(user)
    .map(u -> u.getAddress())
    .map(a -> a.getCountry())
    .map(c -> c.getIsocode())
    .orElse("default");

利用Optional這個jdk8中引入的類就可以優雅的處理,現在我們來詳細講解下這個類的使用和源碼。

創建Optional實例

Optional類, 是對value值進行瞭包裝,它的值可能是null, 也可能不是null,一共有兩個方法創建Optional實例

1.static Optional of(T value)

說明: 傳入的value為null,會拋錯,也就是說value不許不為null

演示:

@Test
public void testOf() {
    Integer value = 2;
    // 正常
    Optional<Integer> op = Optional.of(value);

    value = null;
    // 報空指針
    op = Optional.of(value);
}

源碼:

2.static Optional ofNullable(T value)

說明: 傳入的value可能是null, 也可能不是null

演示:

@Test
    public void testOfNullable() {
        Integer value = 2;
        // 正常
        Optional<Integer> op = Optional.ofNullable(value);
        value = null;
        // 不報錯
        op = Optional.ofNullable(value);
    }

源碼:

獲取Optional中的值

以前創建瞭Optional實例,現在你可能要獲取裡面的value, 有下面幾個方法。

1.T get()

說明: 最樸素的獲取原生value的方法,如果value是空,則直接拋出異常,否則返回。

演示:

@Test
public void testGet() {
    Integer value = 2;
    // 正常
    Optional<Integer> op = Optional.ofNullable(value);
    Integer opVal = op.get();
    Assert.assertEquals(opVal, value);
    
    op = Optional.ofNullable(null);
    // 會拋出異常
    op.get();
}

源碼:

2.T orElse(T other)

說明: 如果值存在返回,否則返回orElse中傳入的other

演示:

@Test
public void testOrElse() {
    // 正常
    Optional<Integer> op = Optional.ofNullable(2);
    Integer opVal = op.orElse(3);
    Assert.assertEquals(opVal, new Integer(2));

    op = Optional.ofNullable(null);
    // 為空,則返回3
    opVal = op.orElse(3);
    Assert.assertEquals(opVal, new Integer(3));
}

源碼:

3.T orElseGet(Supplier<? extends T> other)

說明: 如果存在則返回該值,否則調用other這個函數編程並返回該調用的結果。

演示:

@Test
public void testOrElseGet() {
    // 正常
    Optional<Integer> op = Optional.ofNullable(2);
    Integer opVal = op.orElseGet(()-> {
        return new Integer(3);
    });
    Assert.assertEquals(opVal, new Integer(2));

    op = Optional.ofNullable(null);
    // 為空,則返回3
    opVal = op.orElseGet(()-> {
        return new Integer(3);
    });
    Assert.assertEquals(opVal, new Integer(3));
}

源碼:

問題: orElseGet和orElse有什麼區別嗎?

orElse() 和 orElseGet() 的不同之處在於當 ofNullable() 傳入參數不為空時,orElse() 方法仍然創建瞭 other這個 對象。與之相反,orElseGet() 方法不創建對象。在執行較密集的調用時,比如調用 Web 服務或數據查詢,這個差異會對性能產生重大影響。

而且我們還可以在orElseGet方法中加些日志,可以把這種為空的異常情況暴露出來。

4.T orElseThrow(Supplier<? extends X> exceptionSupplier)

說明: 如果存在則返回該值,否則為空的話可以拋出自定義的異常。

演示:

@Test
public void testOrElseThrow() {
    Optional<Integer> op = Optional.ofNullable(null);
    // 為空,則拋出指定的異常類型
    Integer opVal = op.orElseThrow(()-> {
        return new RuntimeException();
    });
    // 或拋出runtime異常
    Assert.assertEquals(opVal, new Integer(3));
}

源碼:

判斷Optional是否為空

1.boolean isPresent()

說明: 判斷value是否為空

演示:

@Test
public void testIsPresent() {
    Optional<Integer> op = Optional.ofNullable(null);
    // 為空
    Assert.assertFalse(op.isPresent());
}

源碼:

2.void ifPresent(Consumer<? super T> consumer)

說明: 如果存在值,則調用對應的consumer方法,否則不執行任何操作。

演示:

@Test
public void testIfPresent() {
    Optional<Integer> op = Optional.ofNullable(5);
    op.ifPresent(value -> {
        // 如果存在,打印出來
        System.out.println(value);
    });
}

源碼:

Optional中的過濾、轉換方法

此外Optional額外還提供瞭一些過濾、轉換的方法。

1.Optional filter(Predicate<? super T> predicate)

說明: 對Optional中的value進行過濾,如果不匹配,返回空

演示:

@Test
public void testFilter() {
    // 不滿足過濾條件,返回空 Optional
    Optional<String> op = Optional.ofNullable("10").filter(item -> "15".equals(item));
    Assert.assertFalse(op.isPresent());
}

源碼:

2.Optional map(Function<? super T, ? extends U> mapper)

說明: 對Optional中的value進行轉換映射為另外一個對象,如果value為空,返回empty Optional

演示:

源碼:

小結: 回到概述的案例,就是通過不斷的map, 鏈式調用返回內層對象的值。

3.Optional flatMap(Function<? super T, Optional> mapper)

說明: 接受一個返回值為Optional的映射函數參數,該返回值亦是flatMap方法的返回值若結果為空,則返回 空Optional。它也map的區別,我們用一個例子演示出來。

演示:

源碼:

小結: 如果對於返回值非Optional類型,可以用map方法, 否則使用flatMap更加方便

以上就是一文教你搞定Java Optional類判空操作的詳細內容,更多關於Java Optional 判空的資料請關註WalkonNet其它相關文章!

推薦閱讀: