欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java中Optional的正確用法與爭議點詳解

 更新時間:2022年11月07日 08:14:55   作者:打碼日記  
這篇文章主要介紹了Java中Optional的正確用法與爭議點的相關(guān)資料,需要的朋友可以參考下

簡介#

要說Java中什么異常最容易出現(xiàn),我想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("")

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

創(chuàng)建Optional

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

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

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

判空

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

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

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

取值

get()方法獲取Optional的值,當沒有值時會拋出一個NoSuchElementException異常。

同樣的,它也不太應(yīng)該是使用Optional時的第一個使用的方法,因為拋NoSuchElementException與拋NullPointerException并沒有太大區(qū)別。

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

String name = nameOpt.orElse("");

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

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

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

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

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

函數(shù)式處理

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

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

filter方法這個方法用于過濾Optional中的值,若Optional有值,且值滿足過濾函數(shù),則返回此Optional,否則返回空Optional。

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

map方法這個方法用于轉(zhuǎn)換值,在最前面已經(jīng)展示過了,若Optional有值,執(zhí)行map中的lambda函數(shù)轉(zhuǎn)換值,如下:

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

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

Optional爭議點

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

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

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

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

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

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

總結(jié)

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

這和Java的受檢查異常是一樣的,強制要求調(diào)用方處理異常,但又有多少場景的異常是調(diào)用方可以處理的呢?這導致開發(fā)人員經(jīng)常濫用catch,對異常處理一把梭了,最后發(fā)現(xiàn)catch后面還有一些本不該被執(zhí)行的代碼執(zhí)行了。

到此這篇關(guān)于Java中Optional的正確用法與爭議點的文章就介紹到這了,更多相關(guān)Java中Optional用法與爭議點內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Python動態(tài)類型實現(xiàn)原理及過程解析

    Python動態(tài)類型實現(xiàn)原理及過程解析

    這篇文章主要介紹了Python動態(tài)類型實現(xiàn)原理及過程解析,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
    2020-08-08
  • SpringBoot全局配置long轉(zhuǎn)String丟失精度的問題解決

    SpringBoot全局配置long轉(zhuǎn)String丟失精度的問題解決

    web項目中,Java后端傳過來的Long/long類型,前端JS接收會丟失精度。那么應(yīng)該如何解決,本文就來介紹一下幾種方法,感興趣的可以了解一下
    2021-08-08
  • Java實現(xiàn)文件上傳至服務(wù)器的方法

    Java實現(xiàn)文件上傳至服務(wù)器的方法

    這篇文章主要為大家詳細介紹了Java實現(xiàn)文件上傳至服務(wù)器的方法,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2018-01-01
  • SpringSecurity跨域請求偽造(CSRF)的防護實現(xiàn)

    SpringSecurity跨域請求偽造(CSRF)的防護實現(xiàn)

    本文主要介紹了SpringSecurity跨域請求偽造(CSRF)的防護實現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2022-07-07
  • Collections工具類_動力節(jié)點Java學院整理

    Collections工具類_動力節(jié)點Java學院整理

    Collections工具類提供了大量針對Collection/Map的操作。這篇文章主要介紹了Collections工具類_動力節(jié)點Java學院整理,需要的朋友可以參考下
    2017-04-04
  • JAVA實現(xiàn)PDF轉(zhuǎn)HTML文檔的示例代碼

    JAVA實現(xiàn)PDF轉(zhuǎn)HTML文檔的示例代碼

    本文是基于PDF文檔轉(zhuǎn)PNG圖片,然后進行圖片拼接,拼接后的圖片轉(zhuǎn)為base64字符串,然后拼接html文檔寫入html文件實現(xiàn)PDF文檔轉(zhuǎn)HTML文檔,感興趣的可以了解一下
    2021-05-05
  • java實現(xiàn)置換密碼加密解密

    java實現(xiàn)置換密碼加密解密

    這篇文章主要為大家詳細介紹了java實現(xiàn)置換密碼加密解密,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2019-03-03
  • Java web入門指南之在Idea上創(chuàng)建Java web項目

    Java web入門指南之在Idea上創(chuàng)建Java web項目

    好多書上的JavaWeb教程都是Eclipse以及MyEclipse,當然這里不論IDE的好壞,下面這篇文章主要給大家介紹了關(guān)于Java web入門指南之在Idea上創(chuàng)建Java web項目的相關(guān)資料,文中通過實例代碼介紹的非常詳細,需要的朋友可以參考下
    2023-06-06
  • Java Socket實現(xiàn)文件傳輸示例代碼

    Java Socket實現(xiàn)文件傳輸示例代碼

    這篇文章主要介紹了Java Socket實現(xiàn)文件傳輸示例代碼,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-01-01
  • mybatis?plus實現(xiàn)分頁邏輯刪除

    mybatis?plus實現(xiàn)分頁邏輯刪除

    這篇文章主要為大家介紹了mybatis?plus實現(xiàn)分頁邏輯刪除的方式詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-05-05

最新評論