0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫(xiě)文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

JDK8新增的Optional類的常用方法

科技綠洲 ? 來(lái)源:Java技術(shù)指北 ? 作者:Java技術(shù)指北 ? 2023-09-30 15:13 ? 次閱讀

一、摘要

NullPointerException,中文名: 空指針異常 ,也簡(jiǎn)稱 NPE,是軟件系統(tǒng)中最常見(jiàn)的錯(cuò)誤異常之一。

很久以前 Google Guava 項(xiàng)目引入了Optional作為解決空指針異常的一種方式,不贊成寫(xiě)過(guò)多的代碼來(lái)顯式檢查null,以期望程序員寫(xiě)出整潔同時(shí)可讀性更高的代碼。

受 Google Guava 的影響,Optional 現(xiàn)在也成為了Java 8 及以上庫(kù)代碼的一部分。

在介紹Optional技術(shù)之前,我們不禁會(huì)發(fā)出一個(gè)疑問(wèn):為什么谷歌不贊成寫(xiě)過(guò)多的代碼來(lái)顯式檢查null?

下面是某個(gè)常見(jiàn)的參數(shù)判空代碼,樣例如下。

// 判斷行政區(qū)是否為空
if(country != null){
    // 判斷行政區(qū)的上一級(jí),行政城市是否為空
    if(country.getCity() != null){
        // 判斷行政城市的上一級(jí),行政省是否為空
        if(country.getCity().getProvince() != null){
            // 獲取對(duì)應(yīng)的行政省相關(guān)的數(shù)據(jù)
            return country.getCity().getProvince().getName();
        }
    }
}

這還是最普通的三層判斷,假如有很大一段業(yè)務(wù)邏輯處理的時(shí)候,你會(huì)發(fā)現(xiàn)代碼不光看起來(lái)很臃腫,并且難以閱讀,可讀性很差!

如果調(diào)整為使用Optional來(lái)編寫(xiě)的話,可以轉(zhuǎn)換成如下寫(xiě)法:

// 獲取當(dāng)前行政區(qū)最頂級(jí)的省信息名稱
String result  = Optional.ofNullable(country)
                .map(Country::getCity)
                .map(City::getProvince)
                .map(Province::getName)
                .orElse("error");

采用Optional來(lái)編程之后,整個(gè)代碼的可讀性和整潔度,是不是要干凈很多!

這也是為什么推薦大家使用Optional的原因啦!

當(dāng)然廢話也不多說(shuō),代碼直接擼起來(lái)!

二、案例實(shí)踐

在 JDK8 中,Optional 共有 12 個(gè)核心方法,下面我們一起來(lái)看看他們的用法!

2.1、empty()

empty 方法返回一個(gè)不包含值的 Optional 實(shí)例,單獨(dú)使用沒(méi)什么意義,主要和其他方法搭配使用。

Optional optional = Optional.empty();
System.out.println(optional);
-- 輸出結(jié)果
Optional.empty

2.2、of()

of 方法會(huì)返回一個(gè) Optional 實(shí)例,如果傳入的值非空,會(huì)返回包含指定值的對(duì)象;如果傳入空,會(huì)立刻拋出空指針異常。

// 非空情況下,會(huì)正常返回
Optional optional = Optional.of("hello world");
System.out.println(optional);
-- 輸出結(jié)果
Optional[hello world]
// 為空情況下,會(huì)拋空指針異常
Optional optional = Optional.of(null);
System.out.println(optional);
-- 輸出結(jié)果
Exception in thread "main" java.lang.NullPointerException
 at java.util.Objects.requireNonNull(Objects.java:203)
 at java.util.Optional.< init >(Optional.java:96)
 at java.util.Optional.of(Optional.java:108)

2.3、ofNullable()

ofNullable 方法會(huì)返回一個(gè) Optional 實(shí)例,如果傳入的值非空,會(huì)返回包含指定值的對(duì)象;如果傳入空,會(huì)返回不包含任何值的 empty 對(duì)象,也就是最開(kāi)始介紹的Optional.empty()對(duì)象。

// 非空情況下,會(huì)正常返回
Optional optional = Optional.ofNullable("hello world");
System.out.println(optional);
-- 輸出結(jié)果
Optional[hello world]
// 為空情況下,會(huì)返回 empty 對(duì)象
Optional optional = Optional.ofNullable(null);
System.out.println(optional);
-- 輸出結(jié)果
Optional.empty

2.4、isPresent()

isPresent 方法用來(lái)判斷實(shí)例是否包含值,如果包含非空值,返回 true,否則返回 false。

// 非空值,返回true
boolean rs1 =  Optional.ofNullable("hello").isPresent();
System.out.println(rs1);

// 空值,返回false
boolean rs2 =  Optional.ofNullable(null).isPresent();
System.out.println(rs2);
-- 輸出結(jié)果
true
false

2.5、get()

get 方法,如果實(shí)例包含非空值,則返回當(dāng)前值;否則拋出 NoSushElementException 異常。

// 非空值,返回當(dāng)前值
Object rs =  Optional.ofNullable("hello world").get();
System.out.println(rs);
-- 輸出結(jié)果
hello world
// 空值,會(huì)拋出 NoSushElementException 異常
Object rs =  Optional.ofNullable(null).get();
System.out.println(rs);
-- 輸出結(jié)果
Exception in thread "main" java.util.NoSuchElementException: No value present
 at java.util.Optional.get(Optional.java:135)

2.6、ifPresent()

ifPresent 方法作用是當(dāng)實(shí)例包含非空值時(shí),執(zhí)行傳入的 Consumer,比如調(diào)用一些其他方法;如果包含的值為空,不執(zhí)行任何操作。

Optional.ofNullable("hello world")
                .ifPresent( x - > {
                    System.out.println(x);
                });
-- 輸出結(jié)果
hello world

2.7、filter()

filter 方法用于過(guò)濾不符合條件的值,接收一個(gè)Predicate參數(shù),如果符合條件,會(huì)返回當(dāng)前的Optional實(shí)例,否則返回 empty 實(shí)例。

Optional.ofNullable("hello world")
                .filter(x - > x.contains("hello"))
                .ifPresent(x - > {
                    System.out.println(x);
                });
-- 輸出結(jié)果
hello world

2.8、map()

map 方法是鏈?zhǔn)秸{(diào)用避免空指針的核心方法,當(dāng)實(shí)例包含值時(shí),對(duì)值執(zhí)行傳入的Function函數(shù)接口方法,并返回一個(gè)代表結(jié)果值新的Optional實(shí)例,也就是將返回的結(jié)果再次包裝成Optional對(duì)象。

Optional.ofNullable("hello+world")
                .map(t - > {
                    if(t.contains("+")){
                        return t.replace("+", " ");
                    }
                    return t;
                }).ifPresent(t - > {
                    System.out.println(t);
                });
-- 輸出結(jié)果
hello world

2.9、flatMap()

flatMap 方法與 map 方法類似,唯一不同的地方在于: 需要手動(dòng)將返回的值,包裝成Optional實(shí)例,并且參數(shù)值不允許為空 。

Optional.ofNullable("hello+world")
                .flatMap(t - > {
                    if(t.contains("+")){
                        t =  t.replace("+", " ");
                    }
                    // 不同之處
                    return Optional.of(t);
                }).ifPresent(t - > {
                    System.out.println(t);
                });
-- 輸出結(jié)果
hello world

2.10、orElse()

orElse 方法作用是如果實(shí)例包含非空值,那么返回當(dāng)前值;否則返回指定的默認(rèn)值。

Object rs =  Optional.ofNullable(null).orElse("null");
System.out.println(rs);
-- 輸出結(jié)果
null

2.11、orElseGet()

orElseGet 方法作用是如果實(shí)例包含非空值,返回這個(gè)值;否則,它會(huì)執(zhí)行作為參數(shù)傳入的Supplier函數(shù)式接口方法,并返回其執(zhí)行結(jié)果。

Object result = Optional.ofNullable(null)
                .orElseGet(() - > {
                    return "error";
                });
System.out.println(result);
-- 輸出結(jié)果
error

2.12、orElseThrow()

orElseThrow 方法作用是如果實(shí)例包含非空值,返回這個(gè)值;否則,它會(huì)執(zhí)行作為參數(shù)傳入的異常類。

Optional.ofNullable(null)
                .orElseThrow(() - > new RuntimeException("參數(shù)為空"));
-- 輸出結(jié)果
Exception in thread "main" java.lang.RuntimeException: 參數(shù)為空
 at com.x.x.x.x.OptionalTest.lambda$main$10(OptionalTest3.java:144)
 at java.util.Optional.orElseThrow(Optional.java:290)

三、小結(jié)

以上就是 JDK8 新增的Optional類的常用方法總結(jié),其中ofNullable、maporElse方法搭配使用的最多。

另外orElse、orElseGet、orElseThrow區(qū)別如下:

  • orElse:如果實(shí)例包含空值,返回傳入指定的值
  • orElseGet:如果實(shí)例包含空值,返回傳入的方法中返回值
  • orElseThrow:如果實(shí)例包含空值,返回指定的異常類型

在實(shí)際使用的時(shí)候,還得結(jié)合具體的場(chǎng)景進(jìn)行合理選擇,有時(shí)候并不是全部采用Optional來(lái)解決NPE異常代碼才更加優(yōu)雅,比如當(dāng)前對(duì)象比較簡(jiǎn)單,就是一個(gè)簡(jiǎn)單判斷,通過(guò)obj != null足以解決問(wèn)題。

因此在保證業(yè)務(wù)功能的正確和穩(wěn)定性的基礎(chǔ)之上,適當(dāng)?shù)倪x擇相關(guān)的工具來(lái)優(yōu)化代碼的整潔度和可讀性,更能發(fā)揮出錦上添花的效果!

聲明:本文內(nèi)容及配圖由入駐作者撰寫(xiě)或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問(wèn)題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • 程序
    +關(guān)注

    關(guān)注

    117

    文章

    3787

    瀏覽量

    81074
  • 代碼
    +關(guān)注

    關(guān)注

    30

    文章

    4790

    瀏覽量

    68654
  • 軟件系統(tǒng)
    +關(guān)注

    關(guān)注

    0

    文章

    63

    瀏覽量

    9505
  • jdk8
    +關(guān)注

    關(guān)注

    0

    文章

    4

    瀏覽量

    1930
收藏 人收藏

    評(píng)論

    相關(guān)推薦

    JDK動(dòng)態(tài)代理的原理

    在Java中,動(dòng)態(tài)代理是一種機(jī)制,允許在運(yùn)行時(shí)動(dòng)態(tài)地創(chuàng)建代理對(duì)象來(lái)代替某個(gè)實(shí)際對(duì)象,從而在其前后執(zhí)行額外的邏輯。 為什么JDK動(dòng)態(tài)代理只能代理接口實(shí)現(xiàn),原因是JDK動(dòng)態(tài)代理是基于接口實(shí)現(xiàn)的。 當(dāng)你
    的頭像 發(fā)表于 09-30 10:51 ?586次閱讀

    樹(shù)莓派安裝JDK

    /jdk8-downloads-2133151.html最新版本jdk8下載的文件jdk-7u60-linux-arm-vfp-hflt.tar.gz 2、在home下面創(chuàng)建tool文件夾,解壓
    發(fā)表于 03-05 15:12

    畢昇JDK 8 Dynamic CDS 特性介紹

    內(nèi)存中的到 JSA 文件。畢昇JDK 8 中實(shí)現(xiàn)的 Dynamic CDS 特性相比之前的 AppCDS ,增加了 Custom ClassLoader 的支持,擴(kuò)展了共享的支持
    發(fā)表于 12-23 16:19

    JDK8 Optional新特性

    Optional不是對(duì)null關(guān)鍵字的一種替代,而是對(duì)于null判定提供了一種更加優(yōu)雅的實(shí)現(xiàn)。 NullPointException可以說(shuō)是所有java程序員都遇到過(guò)的一個(gè)異常,雖然java從
    發(fā)表于 11-28 14:33 ?1776次閱讀

    Java數(shù)組的常用方法_Java:數(shù)組工具Arrays常用方法的用法及代碼

    本文主要詳細(xì)介紹了Java數(shù)組的常用方法以及數(shù)組工具Arrays常用方法的用法及代碼。
    發(fā)表于 01-29 10:25 ?2925次閱讀

    簡(jiǎn)述那些JDK中坑你沒(méi)商量的方法

    前言 JDK 作為我們每天必備的調(diào)用庫(kù),里面大量提供了基礎(chǔ)供我們使用??梢哉f(shuō)離開(kāi) JDK ,我們的 Java代碼寸步難行。 JDK 帶給
    的頭像 發(fā)表于 06-12 17:36 ?1237次閱讀
    簡(jiǎn)述那些<b class='flag-5'>JDK</b>中坑你沒(méi)商量的<b class='flag-5'>方法</b>

    如何解決JDK8小版本升級(jí)后性能下降的問(wèn)題

    編者按:在升級(jí) JDK8U 的小版本后(從 8u74 升級(jí)到 8u202),遇到性能劇烈下降的問(wèn)題(性能下降 13 倍)。該應(yīng)用是一個(gè)非常簡(jiǎn)單的 Web 應(yīng)用,且應(yīng)用在 JDK 升級(jí)前
    的頭像 發(fā)表于 07-26 14:44 ?4114次閱讀
    如何解決<b class='flag-5'>JDK8</b>小版本升級(jí)后性能下降的問(wèn)題

    畢昇JDK8JDK11首次同時(shí)發(fā)布兩個(gè)版本

    2021 年 9 月 30 日,畢昇 JDK update Q3 版本正式發(fā)布,本次發(fā)布將包含 X86_64 版本。此前,畢昇 JDK 只發(fā)布 Aarch64 版本,這可能會(huì)對(duì)運(yùn)維產(chǎn)生一定
    的頭像 發(fā)表于 10-28 10:53 ?3306次閱讀
    畢昇<b class='flag-5'>JDK8</b>和<b class='flag-5'>JDK</b>11首次同時(shí)發(fā)布兩個(gè)版本

    JAVA8提供了Optional來(lái)優(yōu)化這種寫(xiě)法

    這種寫(xiě)法是比較丑陋的,為了避免上述丑陋的寫(xiě)法,讓丑陋的設(shè)計(jì)變得優(yōu)雅。JAVA8提供了Optional來(lái)優(yōu)化這種寫(xiě)法,接下來(lái)的正文部分進(jìn)行詳細(xì)說(shuō)明
    的頭像 發(fā)表于 04-24 15:18 ?1088次閱讀

    JDK8 Stream數(shù)據(jù)流效率分析

    Stream 是Java SE 8庫(kù)中新增的關(guān)鍵抽象,它被定義于 java.util.stream (這個(gè)包里有若干流類型:Stream 代表對(duì)象引用流,此外還有一系列特化流,如 IntStream,LongStream
    的頭像 發(fā)表于 08-17 10:53 ?1249次閱讀

    基于JDK 1.8來(lái)分析Thread的源碼

    由上圖我們可以看出,Thread實(shí)現(xiàn)了Runnable接口,而Runnable在JDK 1.8中被@FunctionalInterface注解標(biāo)記為函數(shù)式接口,Runnable接口在JDK 1.8中的源代碼如下所示。
    的頭像 發(fā)表于 02-06 17:12 ?634次閱讀

    Java中Arrays是什么 Arrays常用方法

    了解Arrays的概念 **A****rrays** 位于java.util包下,Arrays是一個(gè)操作數(shù)組的工具。 Arrays常用方法 Arrays.fill:
    的頭像 發(fā)表于 02-17 15:11 ?1050次閱讀
    Java中Arrays<b class='flag-5'>類</b>是什么 Arrays<b class='flag-5'>常用</b><b class='flag-5'>方法</b>

    JDK中java.lang.Arrays 的源碼解析

    揭開(kāi)它神秘的面紗。 java.util.Arrays JDK 提供的一個(gè)工具,用來(lái)處理數(shù)組的各種方法,而且每個(gè)方法基本上都是靜態(tài)
    的頭像 發(fā)表于 10-11 15:31 ?618次閱讀
    <b class='flag-5'>JDK</b>中java.lang.Arrays <b class='flag-5'>類</b>的源碼解析

    JDK11升級(jí)JDK17最全實(shí)踐干貨來(lái)了

    1、前言 如果你仍在使用JDK8,那你是否曾經(jīng)遇到過(guò)OutOfMemoryError的問(wèn)題?你是否曾經(jīng)為JVM的調(diào)優(yōu)問(wèn)題感到困擾?本篇文章將為你介紹一種能夠提供百倍性能提升的垃圾回收器,也許能夠
    的頭像 發(fā)表于 06-25 14:50 ?736次閱讀
    <b class='flag-5'>JDK</b>11升級(jí)<b class='flag-5'>JDK</b>17最全實(shí)踐干貨來(lái)了

    JDK8升級(jí)JDK11最全實(shí)踐干貨來(lái)了

    呢?值得我們升級(jí)嗎?而且升級(jí)過(guò)程會(huì)遇到哪些問(wèn)題呢?帶著這些問(wèn)題,本篇文章將帶來(lái)完整的JDK8升級(jí)JDK11最全實(shí)踐。 2、為什么升級(jí)JDK11 1)性能提升 更好的垃圾收機(jī)制、更快的
    的頭像 發(fā)表于 06-25 14:51 ?449次閱讀
    <b class='flag-5'>JDK8</b>升級(jí)<b class='flag-5'>JDK</b>11最全實(shí)踐干貨來(lái)了