告別空指針讓代碼變優雅,Optional使用圖文例子源碼解讀

来源:https://www.cnblogs.com/wang1221/archive/2023/10/18/17771302.html
-Advertisement-
Play Games

一、前言 我們在開發中最常見的異常就是NullPointerException,防不勝防啊,相信大家肯定被坑過! 這種基本出現在獲取資料庫信息中、三方介面,獲取的對象為空,再去get出現! 解決方案當然簡單,只需要判斷一下,不是空在去後續操作,為空返回! 所有在JDK8時出現了專門處理的方案,出來很 ...


一、前言

我們在開發中最常見的異常就是NullPointerException,防不勝防啊,相信大家肯定被坑過!

這種基本出現在獲取資料庫信息中、三方介面,獲取的對象為空,再去get出現!

解決方案當然簡單,只需要判斷一下,不是空在去後續操作,為空返回!

所有在JDK8時出現了專門處理的方案,出來很早了,但是小編慚愧一直沒有去使用它!

最近在看《Java開發手冊》,一直想著提高自己的代碼水平,文中就指出了使用Optional來解決NullPointerException

二、Java開發手冊規範

小編使用的是2022版的黃山版,29頁寫到:

【推薦】防止 NPE,是程式員的基本修養,註意 NPE 產生的場景:

  • 返回類型為基本數據類型,return 包裝數據類型的對象時,自動拆箱有可能產生 NPE

反例:public int method() { return Integer 對象; },如果為 null,自動解箱拋 NPE。

  • 資料庫的查詢結果可能為 null。
  • 集合里的元素即使 isNotEmpty,取出的數據元素也可能為 null。
  • 遠程調用返回對象時,一律要求進行空指針判斷,防止 NPE。
  • 對於 Session 中獲取的數據,建議進行 NPE 檢查,避免空指針。
  • 級聯調用 obj.getA().getB().getC();一連串調用,易產生 NPE。

正例:使用 JDK8 的 Optional 類來防止 NPE 問題。

這份手冊還是不錯的,推薦反覆閱讀,雖然進不去大廠,也要自覺約束自己的代碼風格,努力向大廠靠!

大家現在不知道哪裡找的可以下載一下:

《Java開發手冊》

三、Optional常用方法

小編帶大家一起從api文檔中的方法,一個個帶大家慢慢去瞭解它!

1. empty()

返回一個空的Optional實例:Optional.empty

Optional<Object> empty = Optional.empty();
log.info("empty值:{}",empty);

在這裡插入圖片描述

2. of(T value)

傳入一個參數,返回一個Optional對象,如果參數為空,報NullPointerException

Test testNew  = new Test();
Test test = null;
Optional<Test> optionalNew = Optional.of(testNew);
log.info(" optional對象:{}",optionalNew);
Optional<Test> optional = Optional.of(test);

在這裡插入圖片描述

源碼查看:

我們看到參數為空會報NullPointerException,我們去方法內部看一下就明白了:

public static <T> Optional<T> of(T value) {
    return new Optional<>(value);
}
private Optional(T value) {
    this.value = Objects.requireNonNull(value);
}
public static <T> T requireNonNull(T obj) {
    if (obj == null)
        throw new NullPointerException();
    return obj;
}

我們發現是在Objects類中的requireNonNull方法中判斷了是否為空!

這個還會出現NullPointerException,所以我們一般使用下麵的這個方法!

3. ofNullable(T value)

參數傳入一個對象,返回一個Optional對象,如果為空,將返回一個空的Optional對象,就等於Optional.empty

Test testNew  = new Test();
Test test = null;
Optional<Test> optionalNew = Optional.of(testNew);
log.info(" optional對象:{}",optionalNew);

Optional<Test> optionalTest = Optional.ofNullable(test);
log.info(" optional對象中的ofNullable方法返回值:{}",optionalTest);
Optional<Test> optionalTestNew = Optional.ofNullable(testNew);
log.info(" optional對象中的ofNullable方法new返回值:{}",optionalTestNew);

在這裡插入圖片描述

源碼查看:

我們發現是在方法開始進行非空判斷,再去調用上面的of(T value)方法

public static <T> Optional<T> ofNullable(T value) {
    return value == null ? empty() : of(value);
}

4. get()

如果此Optional中存在值,則返回該值,否則拋出NoSuchElementException

Test testNew  = new Test();
Test test = null;
Optional<Test> optionalNew = Optional.of(testNew);
log.info(" optional對象:{}",optionalNew);
// Optional<Test> optional = Optional.of(test);

Optional<Test> optionalTest = Optional.ofNullable(test);
log.info(" optional對象中的ofNullable方法返回值:{}",optionalTest);
Optional<Test> optionalTestNew = Optional.ofNullable(testNew);
log.info(" optional對象中的ofNullable方法new返回值:{}",optionalTestNew);

Test test2 = optionalTestNew.get();
log.info("原來有值的:經過Optional包裝後get後得到原來的值:{}",test2);
Test test1 = optionalTest.get();
log.info("原來沒有值的:經過Optional包裝後get後得到原來的值:{}",test1);

在這裡插入圖片描述

源碼查看:

調用開始會進行值判斷,如果為空則拋異常!

public T get() {
    if (value == null) {
        throw new NoSuchElementException("No value present");
    }
    return value;
}

5. isPresent()

如果存在值,則返回true,否則返回false。

這裡代碼就不加上面的,大家參考上面的獲取一個Optional對象

boolean present = optionalTestNew.isPresent();
log.info("optionalTestNew調用是否為空:{}",present);
boolean present1 = optionalTest.isPresent();
log.info("optionalTest調用是否為空:{}",present1);

在這裡插入圖片描述

源碼查看:

這就比較簡單了!

public boolean isPresent() {
   return value != null;
}

6. ifPresent(Consumer<? super T> consumer)

如果存在值,則使用該值調用指定的使用者,否則不執行任何操作。

主要的就是入參數一個函數式介面,有值就會去執行,為空則不進行任何操作!

小技巧:

開始對lambda不瞭解時,可以先按照上面這種方式進行寫,

大家可以看到Idea給置灰了,就是可以優化,我們Alt+Enter
然後再次Enter就會變成後面的lambda!

在這裡插入圖片描述

optionalTest.ifPresent(new Consumer<Test>() {
    @Override
    public void accept(Test test) {
        log.info("我是調用ifPresent執行後的列印=====");
    }
});
optionalTestNew.ifPresent(testInner -> log.info("我是調用ifPresent執行後的列印"));

在這裡插入圖片描述

源碼查看:

還是先判斷不為空才去執行函數式介面!

public void ifPresent(Consumer<? super T> consumer) {
    if (value != null)
        consumer.accept(value);
}

7. filter(Predicate<? super T> predicate)

如果存在值,並且該值符合規則,則返回描述該值的Optional,否則返回空Optional

是一個Predicate函數介面,可以傳入實現了Predicate介面的lambda表達式!
如果不符合條件就會返回一個Optional.empty

testNew.setName("蕭炎");
testNew.setAge(33);
Optional<Test> optionalTest1 = optionalTestNew.filter(test1 -> test1.getAge() > 30);
log.info("過濾後的結果:{}",optionalTest1.get());

在這裡插入圖片描述

源碼查看:

就是判斷一下表達式和值是否為空,然後就是根據規則判斷

public Optional<T> filter(Predicate<? super T> predicate) {
   Objects.requireNonNull(predicate);
    if (!isPresent())
        return this;
    else
        return predicate.test(value) ? this : empty();
}

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

如果存在值,則將提供的映射函數應用於該值,如果結果為非空,則返回描述結果的Optional。否則,返回空的Optional。

也是一個函數式介面!

Optional<String> stringOptional = optionalTestNew.map(Test::getName);
log.info("map後獲得欄位值:{}",stringOptional.get());

在這裡插入圖片描述

源碼查看:

也是進行非空判斷,然後執行lambda得到欄位後放到ofNullable方法中!

public<U> Optional<U> map(Function<? super T, ? extends U> mapper) {
    Objects.requireNonNull(mapper);
    if (!isPresent())
        return empty();
    else {
        return Optional.ofNullable(mapper.apply(value));
    }
}

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

如果存在值,則將提供的Optional方位映射函數應用於該值,返回該結果,否則返回空的Optional。此方法類似於map,但提供的映射器的結果已經是可選的,並且如果調用,flatMap不會不會在最後進行任何包裝。

Optional<String> optional = optionalTestNew.flatMap(OptionalTest::getFlatMap);
log.info("flatMap後得到的欄位:{}",optional.get());

private static Optional<String> getFlatMap(Test test){
    return Optional.ofNullable(test).map(Test::getName);
}

在這裡插入圖片描述
源碼查看:

也是進行非空判斷,然後和map不同的是不執行ofNullable方法

public<U> Optional<U> flatMap(Function<? super T, Optional<U>> mapper) {
    Objects.requireNonNull(mapper);
    if (!isPresent())
        return empty();
    else {
        return Objects.requireNonNull(mapper.apply(value));
    }
}

10. orElse(T other)

如果有值則將其返回,否則返回指定的其它值。

如果你是一個對象,orElse()也要是相同對象!

String message = null;
String messageNew = "關註公眾號:小王博客基地";

String nullString = Optional.ofNullable(message).orElse("這是一個空字元串!");
log.info("這是空字元串列印的:{}",nullString);
String string = Optional.ofNullable(messageNew).orElse("=====這是一個空字元串!");
log.info("這是字元串列印的:{}",string);

在這裡插入圖片描述

源碼查看:

簡單的為空返回自己定義的,不為空直接返回!

public T orElse(T other) {
    return value != null ? value : other;
}

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

返回值(如果存在),否則調用other並返回該調用的結果。

區別:
orElse方法將傳入的參數作為預設值,orElseGet方法可以接受Supplier介面的實現用來生成預設值

如果沒有複雜操作,Idea也會提醒我們不要使用這個,使用orElse即可!

String message = null;
String messageNew = "關註公眾號:小王博客基地";
String orElseGet = Optional.ofNullable(message).orElseGet(() -> "這還是一個空的字元串");
log.info("orElseGet調用:這是空字元串列印的:{}",orElseGet);
String orElseGetString = Optional.ofNullable(messageNew).orElseGet(() -> "這還是一個空的字元串");
log.info("orElseGet調用:這是字元串列印的:{}",orElseGetString);

在這裡插入圖片描述

源碼查看:

和orElse一樣,只不過為空調用lambda執行!

public T orElseGet(Supplier<? extends T> other) {
    return value != null ? value : other.get();
}

12. orElseThrow(Supplier<? extends X> exceptionSupplier)

返回包含的值(如果存在),否則拋出由提供的供應商創建的異常。

String message = null;
String messageNew = "關註公眾號:小王博客基地";
Optional.ofNullable(messageNew).orElseThrow(() -> new RuntimeException("為空了,還不看看!"));
Optional.ofNullable(message).orElseThrow(() -> new RuntimeException("為空了,還不看看!"));

我們可以自定義異常,然後來引用!

在這裡插入圖片描述

源碼查看:

為空則走自己寫的異常!

public <X extends Throwable> T orElseThrow(Supplier<? extends X> exceptionSupplier) throws X {
    if (value != null) {
        return value;
    } else {
        throw exceptionSupplier.get();
    }
}

13. 例子彙總

/**
 * @author wangzhenjun
 * @date 2023/2/27 10:22
 */
@Slf4j
public class OptionalTest {

    public static void main(String[] args) {

        Optional<Object> empty = Optional.empty();
        log.info("empty值:{}",empty);


        Test testNew  = new Test();
        Test test = null;
        Optional<Test> optionalNew = Optional.of(testNew);
        log.info(" optional對象:{}",optionalNew);
//        Optional<Test> optional = Optional.of(test);

        Optional<Test> optionalTest = Optional.ofNullable(test);
        log.info(" optional對象中的ofNullable方法返回值:{}",optionalTest);
        Optional<Test> optionalTestNew = Optional.ofNullable(testNew);
        log.info(" optional對象中的ofNullable方法new返回值:{}",optionalTestNew);

        Test test2 = optionalTestNew.get();
        log.info("原來有值的:經過Optional包裝後get後得到原來的值:{}",test2);
        // Test test1 = optionalTest.get();
        // log.info("原來沒有值的:經過Optional包裝後get後得到原來的值:{}",test1);

        boolean present = optionalTestNew.isPresent();
        log.info("optionalTestNew調用是否為空:{}",present);
        boolean present1 = optionalTest.isPresent();
        log.info("optionalTest調用是否為空:{}",present1);

        optionalTest.ifPresent(new Consumer<Test>() {
            @Override
            public void accept(Test test) {
                log.info("我是調用ifPresent執行後的列印=====");
            }
        });
        optionalTestNew.ifPresent(testInner -> log.info("我是調用ifPresent執行後的列印"));

        testNew.setName("蕭炎");
        testNew.setAge(33);
        Optional<Test> optionalTest1 = optionalTestNew.filter(test1 -> test1.getAge() > 30);
        log.info("過濾後的結果:{}",optionalTest1.get());

        Optional<String> stringOptional = optionalTestNew.map(Test::getName);
        log.info("map後獲得欄位值:{}",stringOptional.get());

        Optional<String> optional = optionalTestNew.flatMap(OptionalTest::getFlatMap);
        log.info("flatMap後得到的欄位:{}",optional.get());

        String message = null;
        String messageNew = "關註公眾號:小王博客基地";

        String nullString = Optional.ofNullable(message).orElse("這是一個空字元串!");
        log.info("這是空字元串列印的:{}",nullString);
        String string = Optional.ofNullable(messageNew).orElse("=====這是一個空字元串!");
        log.info("這是字元串列印的:{}",string);

        String orElseGet = Optional.ofNullable(message).orElseGet(() -> "這還是一個空的字元串");
        log.info("orElseGet調用:這是空字元串列印的:{}",orElseGet);
        String orElseGetString = Optional.ofNullable(messageNew).orElseGet(() -> "這還是一個空的字元串");
        log.info("orElseGet調用:這是字元串列印的:{}",orElseGetString);

        Optional.ofNullable(messageNew).orElseThrow(() -> new RuntimeException("為空了,還不看看!"));
        Optional.ofNullable(message).orElseThrow(() -> new RuntimeException("為空了,還不看看!"));



    }

    private static Optional<String> getFlatMap(Test test){
        return Optional.ofNullable(test).map(Test::getName);
    }

}

四、總結

這裡就不在演示實戰了,基本上組合使用:

Optional.ofNullable(需要判斷的對象).ifPresent(具體操作)

其實和if相比就是顯得優雅一些,主要是防止某處沒考慮到,忘記if判斷,那麼後續可能會導致空指針,如果使用Optional的話,那麼這個問題能夠得到避免。

就像多使用設計模式一樣,讓自己的代碼更加健壯優雅,還是要多使用一些的!當然不能過渡使用!!

對你有幫助,還請不要吝嗇你的發財小手點點關註哈!、
寫作不易,大家給點支持,你的支持是我寫作的動力哈!

關註小編的微信公眾號:『小王博客基地』,一起交流學習!文章首發看哦!

建了一個IT交流群,歡迎大家加入,過期加我拉你們進哈!


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 在 Python 編程中,異常是一種常見的情況,可能會導致程式中斷或產生錯誤。然而,並非所有的異常都需要立即處理,有時候我們希望忽略某些異常並繼續執行程式。本文將介紹如何在 Python 中忽略異常,並提供一些示例和註意事項。 try-except 塊: 在 Python 中,我們可以使用 try- ...
  • 前言 ElasticSearch Java API是ES官方在8.x版本推出的新java api,也可以適用於7.17.x版本的es。 本文主要參考了相關博文,自己手動編寫了下相關操作代碼,包括更新mappings等操作的java代碼。 代碼示例已上傳github。 版本 elasticsearch ...
  • GCC編譯 預處理->編譯->彙編->鏈接 預處理:頭⽂件包含、巨集替換、條件編譯、刪除註釋... 編譯:主要進⾏詞法、語法、語義分析等,檢查⽆誤後將預處理好的⽂件編譯成彙編⽂件... 彙編:將彙編⽂件轉換成 ⼆進位⽬標⽂件... 鏈接:將項⽬中的各個⼆進位⽂件+所需的庫+啟動代碼鏈接成可執⾏⽂件.. ...
  • IOI2018 werewolf 狼人 題解 題目描述 省流: \(n\) 個點,\(m\) 條邊,\(q\) 次詢問,對於每一次詢問,給定一個起點 \(S\) 和終點 \(T\) ,能否找到一條路徑,前半程不能走 \(0\thicksim L-1\) 這些點,後半程不能走 \(R+1\thicks ...
  • 基於java線上家政預約服務系統設計與實現,可適用於java家政服務系統,java預約家政系統,java線上家政系統,線上服務系統,社會家政系統,家政管理系統,家政服務平臺,家政更加服務平臺系統,家政管理系統等等; ...
  • 在本文中,我們將全面深入地探討Go語言的反射機制。從反射的基礎概念、為什麼需要反射,到如何在Go中實現反射,以及在高級編程場景如泛型編程和插件架構中的應用,本文為您提供一站式的學習指南。 關註【TechLeadCloud】,分享互聯網架構、雲服務技術的全維度知識。作者擁有10+年互聯網服務架構、AI ...
  • 散點圖,又名點圖、散佈圖、X-Y圖,是將所有的數據以點的形式展現在平面直角坐標繫上的統計圖表。 散點圖常被用於分析變數之間的相關性。如果兩個變數的散點看上去都在一條直線附近波動,則稱變數之間是線性相關的;如果所有點看上去都在某條曲線(非直線)附近波動,則稱此相關為非線形相關的;如果所有點在圖中沒有顯 ...
  • 一、前言 你有看到過那種不間斷型的、迴圈播放視頻音樂的直播間嗎?或者那種直播播放電影的直播間?還有層出不窮的文章,類似如下標題: “如何搭建一個24小時不間斷的直播間?躺入xxxx元!” “24小時電影直播間,每天到賬xxx~xxxx,不出鏡副業,人人可做!” “50塊的雲伺服器直播推流讓我月入過千 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...