Optional用法與爭議點

来源:https://www.cnblogs.com/codelogs/archive/2022/11/06/16863888.html
-Advertisement-
Play Games

原創:扣釘日記(微信公眾號ID:codelogs),歡迎分享,轉載請保留出處。 簡介 要說Java中什麼異常最容易出現,我想NullPointerException一定當仁不讓,為瞭解決這種null值判斷問題,Java8中提供了一個新的工具類Optional,用於提示程式員註意null值,併在特定場 ...


原創:扣釘日記(微信公眾號ID:codelogs),歡迎分享,轉載請保留出處。

簡介

要說Java中什麼異常最容易出現,我想NullPointerException一定當仁不讓,為瞭解決這種null值判斷問題,Java8中提供了一個新的工具類Optional,用於提示程式員註意null值,併在特定場景中簡化代碼邏輯。

比如下麵一段取深層屬性值的代碼:

Order order = getOrderById(orderId);
String userCode = "";
if(order != null){
    if(order.getUser() != null){
        if(order.getUser().getUserCode() != null){
            userCode = order.getUser().getUserCode().toUpperCase();
        }
    }
}

這種場景還比較常見,但深層嵌套的if判斷,讓代碼閱讀者壓力倍增。

看看用Optional後的寫法,如下:

Order order = getOrderById(orderId);
String userCode = Optional.ofNullable(order)
    .map(Order::getUser)
    .map(User::getUserCode)
    .map(String::toUpperCase)
    .orElse("")

鏈式調用的寫法,讓代碼可讀性增強了不少,不用判斷null,是因為Optional在內部已經做了null值判斷了!那我們來看看Optional都有哪些用法吧。

創建Optional

ofNullable()方法
創建一個Optional,傳入的值可以是null或不是null。

of()方法
得到一個Optional,明確知道傳入的值不是null時用這個,如果傳null會報錯NullPointerExcepiton。
其實值不為null一般是沒必要使用Optional的,這個應該是用於特殊場景,比如方法返回值必須是一個Optional。

empty()方法
得到一個空的Optional,一般也用於返回值必須是Optional的場景。

判空

ifPresent()方法
判斷是否有值,註意,這個方法雖然看起來挺好用的,但它不太應該是使用Optional時第一個使用的方法,如下:

if(opt.ifPresent()){
    ...
}
if(obj != null) {
    ...
}

這兩個代碼除了寫法不一樣,對於代碼可讀性方面沒有根本區別!

取值

get()方法
獲取Optional的值,當沒有值時會拋出一個NoSuchElementException異常。
image_2022-11-06_20221106180845
同樣的,它也不太應該是使用Optional時的第一個使用的方法,因為拋NoSuchElementException與拋NullPointerException並沒有太大區別。

orElse方法
沒有值時會返回指定的值。

String name = nameOpt.orElse("");

orElseGet方法
同上,不過參數變成了一個提供預設值的lambda函數,這用在取指定值需要一定代價的場景,如下:

BigDecimal amount = amountOpt.orElseGet(() -> calcDefaultAmount());

orElseThrow方法
沒有值時拋出一個指定的異常,如下:

Optional<User> userOpt = getUser(userId);
User user = userOpt.orElseThrow(() -> new NullPointerException("userId:" + userId));

可能會有人疑問,你還是拋出了一個NPE異常,和不使用Optional有啥區別?區別是這個NPE異常會告訴你哪個userId查不到數據,方便定位問題,而jvm拋出的NPE是沒有這個信息的。

函數式處理

ifPresent(Consumer<? super T> consumer)方法
這個方法和ifPresent()方法不一樣,這個方法代表如果Optional有值時,就執行傳入的lambda函數,如下:

userOpt.ifPresent((user) -> System.out.printf("%s\n", user.toString()));

filter方法
這個方法用於過濾Optional中的值,若Optional有值,且值滿足過濾函數,則返回此Optional,否則返回空Optional。
image_2022-11-06_20221106200230

String name = nameOpt.filter(StringUtils::isNotBlank).orElse("");

map方法
這個方法用於轉換值,在最前面已經展示過了,若Optional有值,執行map中的lambda函數轉換值,如下:

Order order = getOrderById(orderId);
String userCode = Optional.ofNullable(order)
    .map(Order::getUser)
    .map(User::getUserCode)
    .map(String::toUpperCase)
    .orElse("")

Optional還提供了一個flatMap方法,與map方法的區別是,傳給flatMap的lambda函數,這個lambda函數的返回值需要是Optional。

Optional爭議點

其實到底該不該用Optional,業界還是有不少爭議的,一方面是Optional能強迫開發者處理null值,但另一方面是Optional又非常容易濫用,特別是一些開發者拿到Optional之後就直接調用get()ifPresent()方法,這樣幾乎沒解決任何問題,還加重了編碼負擔。

因此,我的建議是,在你不知道該不該使用Optional的場景,那就先別用。

下麵是一些使用Optional的場景參考,如下:

  1. Optional一般用於返回值
    Optional大多用於返回值,不推薦用在成員變數或方法參數中。
  2. Optional本身不判null
    永遠都不要給Optional賦值null,也不要判斷Optional本身是否為null,這是因為Optional本來就是解決null的,再引入null就沒意思了,這應該成為業界共識。
  3. 集合不使用Optional
    因為集合有Collections.emptyList()等更好的處理方法了,沒必要再使用Optional。
  4. 函數式處理值
    從上面的用法介紹中就能發現,Optional提供了很多lambda函數式處理的方法,如filter、map等,這些是使用Optional時比較推薦使用的,因為Optional能幫你自動處理null值情況,避免NPE異常。
  5. 多層屬性獲取
    前面舉過這個代碼樣例,我覺得這是Optional使用收益最明顯的一個場景。
  6. 不返回null勝過返回Optional
    返回Optional給調用方,會強制調用方處理null情況,會給調用方增加一些的編碼負擔,特別是復用度很高的函數。
    但如果調用方大多數情況下都不期望獲取到null,那應該實現一個這樣的方法,要麼返回值,要麼異常,如下:
/**
 * 查詢訂單,要麼返回訂單,要麼異常
 */
public Order getOrderByIdOrExcept(Long orderId){
    Order order = orderMapper.getOrderById(orderId);
    if(order == null){
        throw new BizException("根據單號" + orderId + "未查詢到訂單!");
    }
    return order;
}

/**
 * 查詢訂單,值可能為null
 */
public Optional<Order> getOrderById(Long orderId){
    Order order = orderMapper.getOrderById(orderId);
    return Optional.ofNullable(order);
}

由於後面處理代碼依賴訂單數據,獲取不到訂單數據,代碼也沒法往下走,所以在大多數情況下,選擇使用getOrderByIdOrExcept方法更好,即避免了NPE,又避免了增加編碼負擔!

總結

Optional能解決一些問題,但因為容易濫用而爭議很大,因為Optional將null的處理交給調用方了,但大多數情況下,調用方也沒辦法處理這個null情況,還不如讓JVM拋一個NPE異常中止執行,因為如果你用ifPresent的話,還更容易造成邏輯bug導致執行了不該執行的代碼。

這和Java的受檢查異常是一樣的,強制要求調用方處理異常,但又有多少場景的異常是調用方可以處理的呢?這導致開發人員經常濫用catch,對異常處理一把梭了,最後發現catch後面還有一些本不該被執行的代碼執行了。


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

-Advertisement-
Play Games
更多相關文章
  • 您好,我是湘王,這是我的博客園,歡迎您來,歡迎您再來~ Lambda表達式雖然將介面作為代碼塊賦值給了變數,但如果僅僅只是Lambda表達式,還無法讓Java由量變引起質變。真正讓Lambda能夠發揮出巨大威力的,就是流式計算。 所謂流式計算,就是讓數據像在流水線上一樣,從一道工序流轉到下一道工序。 ...
  • 簡介: 橋接模式又叫橋梁模式,屬於結構型模式。目的是將抽象與實現分離,使它們都可以獨立的變化,解耦。繼承有很多好處,但是會增加耦合,而橋接模式偏向組合和聚合的方式來共用。 適用場景: 不希望或不適用使用多繼承的場景。 一個類存在2個或更多的 獨立變化維度 , 並且這些維度都需要 獨立擴展 優點: 解 ...
  • 日誌概念 1. 日誌文件 日誌文件是用於記錄系統操作事件的文件集合 1.1 調試日誌 1.2 系統日誌 系統日誌是記錄系統中硬體、軟體和系統問題的信息,同時還可以監視系統中發生的事件。用戶可以通過它來檢查錯誤發生的原因,或者尋找受到攻擊時攻擊者留下的痕跡 日誌門面 當我們的系統變的更加複雜的時候,我 ...
  • Fsm1 這裡需要實現一個簡單的摩爾狀態機,即輸出只與狀態有關的狀態機。 我這裡代碼看上去比長一點,答案用的case和三目運算符,結果是一樣的。 module top_module( input clk, input areset, // Asynchronous reset to state B ...
  • 看《C++ Primer Plus》時整理的學習筆記,部分內容完全摘抄自《C++ Primer Plus》(第6版)中文版,Stephen Prata 著,張海龍 袁國忠譯,人民郵電出版社。只做學習記錄用途。 ...
  • 前言 ​ 對於我們平時寫代碼運行,我們很少去關註編譯和鏈接的過程,因為現在的開發環境都是集成(IDE)的,這些IDE一般都會將編譯和鏈接的過程一步搞定,這一過程又被稱為構建。但若經常寫代碼,經常會有很多莫名其妙的錯誤讓我們不知所措,對於這些錯誤若我們能知其原因,那是再好不過了。因此本系列就是帶你瞭解 ...
  • C++ 類:實體的抽象類型 實體(屬性,行為) ->ADT(abstract data type) 類(屬性->成員變數,行為->成員方法) OOP語言4大特征 抽象 封裝/隱藏(通過public private protected) 繼承 多態 點擊查看代碼 class Student{ //屬性 ...
  • 免費課頁面前端搭建 點擊查看代碼 <template> <div class="course"> <Header></Header> <div class="main"> <!-- 篩選條件 --> <div class="condition"> <ul class="cate-list"> <li ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...