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

Java8中新判空方法之Optional類的使用詳解

 更新時間:2022年12月30日 09:24:06   作者:南北?  
Opitonal類就是Java提供的為了解決大家平時判斷對象是否為空用的。本文將通過示例為大家講解一下Optional類的使用,感興趣的可以收藏一下

認(rèn)識Optional

Opitonal類就是Java提供的為了解決大家平時判斷對象是否為空用,通常會用 null!=obj 這樣的方式存在的判斷,從而令人頭疼導(dǎo)致空指針異常,同Optional的存在可以讓代碼更加簡單,可讀性跟高,代碼寫起來更高效

Student?student?=?new?Student();
if?(null?==?student){
????return?"student為null";
}
return?student;
Student?student?=?new?Student();
return?Optional.ofNullable(student).orElse("student為null");

測試展示類Student 代碼(如果有朋友不明白可以看一下這個):

@Data
@AllArgsConstructor
@NoArgsConstructor
public?class?Student?{
????private?String?name;
????private?Integer?age;
}

Optional對象創(chuàng)建

首先我們先打開Optional的內(nèi)部,去一探究竟 先把幾個創(chuàng)建Optional對象的方法提取出來

public?final?class?Optional<T>?{
???private?static?final?Optional<?>?EMPTY?=?new?Optional<>();
???private?final?T?value;
???//?我們可以看到兩個構(gòu)造方格都是private?私有的
???//?說明?我們沒辦法在外面去new出來Optional對象
???private?Optional()?{
????????this.value?=?null;
????}
???private?Optional(T?value)?{
????????this.value?=?Objects.requireNonNull(value);
????}
????//?這個靜態(tài)方法大致?是創(chuàng)建出一個包裝值為空的一個對象因?yàn)闆]有任何參數(shù)賦值
???public?static<T>?Optional<T>?empty()?{
????????@SuppressWarnings("unchecked")
????????Optional<T>?t?=?(Optional<T>)?EMPTY;
????????return?t;
????}
????//?這個靜態(tài)方法大致?是創(chuàng)建出一個包裝值非空的一個對象?因?yàn)樽隽速x值
???public?static?<T>?Optional<T>?of(T?value)?{
????????return?new?Optional<>(value);
????}
????//?這個靜態(tài)方法大致是?如果參數(shù)value為空,則創(chuàng)建空對象,如果不為空,則創(chuàng)建有參對象
???public?static?<T>?Optional<T>?ofNullable(T?value)?{
????????return?value?==?null???empty()?:?of(value);
????}
?}

再做一個簡單的實(shí)例展示 與上面對應(yīng)

//?1、創(chuàng)建一個包裝對象值為空的Optional對象
Optional<String>?optEmpty?=?Optional.empty();
//?2、創(chuàng)建包裝對象值非空的Optional對象
Optional<String>?optOf?=?Optional.of("optional");
//?3、創(chuàng)建包裝對象值允許為空也可以不為空的Optional對象
Optional<String>?optOfNullable1?=?Optional.ofNullable(null);
Optional<String>?optOfNullable2?=?Optional.ofNullable("optional");

Optional.get()方法(返回對象的值)

get()方法是返回一個option的實(shí)例值

源碼:

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

也就是如果value不為空則做返回,如果為空則拋出異常 “No value present” 簡單實(shí)例展示

Student?student?=?new?Student();
student.setAge(18);
System.out.println(Optional.ofNullable(student).get());

Optional.isPresent()方法(判讀是否為空)

isPresent()方法就是會返回一個boolean類型值,如果對象不為空則為真,如果為空則false

源碼:

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

簡單的實(shí)例展示:

Student?student?=?new?Student();
student.setAge(18);
if?(Optional.ofNullable(student).isPresent()){
????System.out.println("不為空");
}else?{
????System.out.println("為空");
}

Optional.ifPresent()方法(判讀是否為空并返回函數(shù))

這個意思是如果對象非空,則運(yùn)行函數(shù)體

源碼:

public?void?ifPresent(Consumer<??super?T>?consumer)?{
????//如果value不為空,則運(yùn)行accept方法體
????if?(value?!=?null)
????????consumer.accept(value);
}

看實(shí)例:

Student?student?=?new?Student();
student.setAge(18);
Optional.ofNullable(student).ifPresent(s?->?System.out.println("年齡:"?+?s.getAge()));

如果對象不為空,則會打印這個年齡,因?yàn)閮?nèi)部已經(jīng)做了NPE(非空判斷),所以就不用擔(dān)心空指針異常了

Optional.filter()方法(過濾對象)

filter()方法大致意思是,接受一個對象,然后對他進(jìn)行條件過濾,如果條件符合則返回Optional對象本身,如果不符合則返回空Optional

源碼:

public?Optional<T>?filter(Predicate<??super?T>?predicate)?{
????Objects.requireNonNull(predicate);
????//如果為空直接返回this
????if?(!isPresent())
????????return?this;
????else
????????//判斷返回本身還是空Optional
????????return?predicate.test(value)???this?:?empty();
}

簡單實(shí)例:

Student?student?=?new?Student();
student.setAge(18);
Optional.ofNullable(student).filter(s?->?s.getAge()>10);

Optional.flatMap()方法(Optional對象進(jìn)行二次包裝)

map()方法將對應(yīng)Optional< Funcation >函數(shù)式接口中的對象,進(jìn)行二次運(yùn)算,封裝成新的對象然后返回在Optional中

源碼:

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));
????}
}

實(shí)例:

Student?student?=?new?Student();
student.setAge(18);
Optional<Object>?optName?=?Optional.ofNullable(student).map(s?->?Optional.ofNullable(s.getName()).orElse("name為空"));

Optional.orElse()方法(為空返回對象)

常用方法之一,這個方法意思是如果包裝對象為空的話,就執(zhí)行orElse方法里的value,如果非空,則返回寫入對象

源碼:

public?T?orElse(T?other)?{
????//如果非空,返回value,如果為空,返回other
????return?value?!=?null???value?:?other;
}

實(shí)例:

Student?student?=?new?Student();
student.setAge(18);
Optional.ofNullable(student).orElse(new?Student("小明",?12));

Optional.orElseGet()方法(為空返回Supplier對象)

這個與orElse很相似,入?yún)⒉灰粯?,入?yún)镾upplier對象,為空返回傳入對象的.get()方法,如果非空則返回當(dāng)前對象。

源碼:

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

實(shí)例:

Optional<Supplier<Student>>?sup=Optional.ofNullable(Student::new);
//調(diào)用get()方法,此時才會調(diào)用對象的構(gòu)造方法,即獲得到真正對象
Optional.ofNullable(student).orElseGet(sup.get());

Suppiler是一個接口,是類似Spring的懶加載,聲明之后并不會占用內(nèi)存,只有執(zhí)行了get()方法之后,才會調(diào)用構(gòu)造方法創(chuàng)建出對象 創(chuàng)建對象的語法的話就是Supplier supStudent= Student::new; 需要使用時supStudent.get()即可

Optional.orElseThrow()方法(為空返回異常)

方法作用的話就是如果為空,就拋出你定義的異常,如果不為空返回當(dāng)前對象,在實(shí)戰(zhàn)中所有異??隙ㄊ且幚砗玫?,為了代碼的可讀性

源碼:

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

實(shí)例:

//簡單的一個查詢
Member?member?=?memberService.selectByPhone(request.getPhone());
Optional.ofNullable(member).orElseThrow(()?->?new?ServiceException("沒有查詢的相關(guān)數(shù)據(jù)"));

相似方法進(jìn)行對比分析

可能看到這,沒用用過的話會覺得orElse()orElseGet()還有orElseThrow()很相似,map()和flatMap()好相似,不用著急,都是從這一步過來的,我再給大家總結(jié)一下不同方法的異同點(diǎn) orElse()orElseGet()orElseThrow()的異同點(diǎn)

方法效果類似,如果對象不為空,則返回對象,如果為空,則返回方法體中的對應(yīng)參數(shù),所以可以看出這三個方法體中參數(shù)是不一樣的 orElse(T 對象) orElseGet(Supplier < T >對象) orElseThrow(異常)

map()和orElseGet的異同點(diǎn)

方法效果類似,對方法參數(shù)進(jìn)行二次包裝,并返回,入?yún)⒉煌?map(function函數(shù)) flatmap(Optional< function >函數(shù))

具體要怎么用,要根據(jù)業(yè)務(wù)場景以及代碼規(guī)范來定義,下面可以簡單看一下我在實(shí)戰(zhàn)中怎用使用神奇的Optional。

實(shí)戰(zhàn)場景

場景1:在service層中 查詢一個對象,返回之后判斷是否為空并做處理

//查詢一個對象
Member?member?=?memberService.selectByIdNo(request.getCertificateNo());
//使用ofNullable加orElseThrow做判斷和操作
Optional.ofNullable(member).orElseThrow(()?->?new?ServiceException("沒有查詢的相關(guān)數(shù)據(jù)"));

場景2:我們可以在dao接口層中定義返回值時就加上Optional

public?interface?LocationRepository?extends?JpaRepository<Location,?String>?{
Optional<Location>?findLocationById(String?id);
}

然在是Service中

public?TerminalVO?findById(String?id)?{
????//這個方法在dao層也是用了Optional包裝了
????Optional<Terminal>?terminalOptional?=?terminalRepository.findById(id);
????//直接使用isPresent()判斷是否為空
????if?(terminalOptional.isPresent())?{
????//使用get()方法獲取對象值
????????Terminal?terminal?=?terminalOptional.get();
????????//在實(shí)戰(zhàn)中,我們已經(jīng)免去了用set去賦值的繁瑣,直接用BeanCopy去賦值
????????TerminalVO?terminalVO?=?BeanCopyUtils.copyBean(terminal,?TerminalVO.class);
????????//調(diào)用dao層方法返回包裝后的對象
????????Optional<Location>?location?=?locationRepository.findLocationById(terminal.getLocationId());
????????if?(location.isPresent())?{
????????????terminalVO.setFullName(location.get().getFullName());
????????}
????????return?terminalVO;
????}
????//不要忘記拋出異常
????throw?new?ServiceException("該終端不存在");
}

實(shí)戰(zhàn)場景還有很多,包括return時可以判斷是否返回當(dāng)前值還是跳轉(zhuǎn)到另一個方法體中,其它的還有很多。

Optional使用注意事項(xiàng)

Optional真么好用,真的可以完全替代if判斷嗎?我想這肯定是大家使用完之后Optional之后可能會產(chǎn)生的想法,答案是否定的 舉一個最簡單的栗子:

例子:如果我只想判斷對象的某一個變量是否為空并且做出判斷呢?

Person?person=new?Person();
person.setName("");
persion.setAge(2);
//普通判斷
if(StringUtils.isNotBlank(person.getName())){
???//名稱不為空執(zhí)行代碼塊
}
//使用Optional做判斷
Optional.ofNullable(person).map(p?->?p.getName()).orElse("name為空");

我覺得這個例子就能很好的說明這個問題,只是一個很簡單判斷,如果用了Optional我們還需要考慮包裝值,考慮代碼書寫,考慮方法調(diào)用,雖然只有一行,但是可讀性并不好,如果別的程序員去讀,我覺得肯定沒有if看的明顯。

Jdk 9對Optional優(yōu)化

首先增加了三個方法: or()、ifPresentOrElse() 和 stream()。or() 與orElse等方法相似,如果對象不為空返回對象,如果為空則返回or()方法中預(yù)設(shè)的值。

ifPresentOrElse() 方法有兩個參數(shù):一個 Consumer 和一個 Runnable。如果對象不為空,會執(zhí)行 Consumer 的動作,否則運(yùn)行 Runnable。相比ifPresent()多了OrElse判斷。stream()將Optional轉(zhuǎn)換成stream,如果有值就返回包含值的stream,如果沒值,就返回空的stream。

以上就是Java8中新判空方法之Optional類的使用詳解的詳細(xì)內(nèi)容,更多關(guān)于Java8 Optional類的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • Java注解使用及原理解析

    Java注解使用及原理解析

    這篇文章主要介紹了Java注解使用及原理解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2020-06-06
  • SpringBoot深入理解之內(nèi)置web容器及配置的總結(jié)

    SpringBoot深入理解之內(nèi)置web容器及配置的總結(jié)

    今天小編就為大家分享一篇關(guān)于SpringBoot深入理解之內(nèi)置web容器及配置的總結(jié),小編覺得內(nèi)容挺不錯的,現(xiàn)在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧
    2019-03-03
  • 淺談Java類的加載,鏈接及初始化

    淺談Java類的加載,鏈接及初始化

    今天給大家?guī)淼氖顷P(guān)于Java的相關(guān)知識,文章圍繞著Java類的加載,鏈接及初始化展開,文中有非常詳細(xì)的解釋及代碼示例,需要的朋友可以參考下
    2021-06-06
  • SpringBoot實(shí)現(xiàn)接口文檔自動生成的方法示例

    SpringBoot實(shí)現(xiàn)接口文檔自動生成的方法示例

    在開發(fā)Web應(yīng)用程序時,接口文檔是非常重要的一環(huán),本文主要介紹了SpringBoot實(shí)現(xiàn)接口文檔自動生成的方法示例,具有一定的參考價值,感興趣的可以了解一下
    2023-10-10
  • MyBatis-Plus如何通過注解使用TypeHandler

    MyBatis-Plus如何通過注解使用TypeHandler

    這篇文章主要介紹了MyBatis-Plus如何通過注解使用TypeHandler,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-01-01
  • Java基礎(chǔ)之List內(nèi)元素的排序性能對比

    Java基礎(chǔ)之List內(nèi)元素的排序性能對比

    這篇文章主要介紹了Java基礎(chǔ)之List內(nèi)元素的排序性能對比,文中有非常詳細(xì)的代碼示例,對正在學(xué)習(xí)java基礎(chǔ)的小伙伴們有非常好的幫助,需要的朋友可以參考下
    2021-04-04
  • Java運(yùn)算符解密之位運(yùn)算、移位運(yùn)算舉例詳解

    Java運(yùn)算符解密之位運(yùn)算、移位運(yùn)算舉例詳解

    這篇文章主要介紹了Java運(yùn)算符解密之位運(yùn)算、移位運(yùn)算的相關(guān)資料,Java中的位運(yùn)算符包括按位與&、按位或|、按位取反~和按位異或^,用于對數(shù)據(jù)的二進(jìn)制位進(jìn)行操作,文中通過代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2025-04-04
  • 詳解spring mvc4使用及json 日期轉(zhuǎn)換解決方案

    詳解spring mvc4使用及json 日期轉(zhuǎn)換解決方案

    本篇文章主要介紹了spring mvc4使用及json 日期轉(zhuǎn)換解決方案,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-01-01
  • JavaScript不使用臨時變量交換兩個變量值的方法總結(jié)

    JavaScript不使用臨時變量交換兩個變量值的方法總結(jié)

    在?JavaScript?中交換兩個變量的值是一項(xiàng)基礎(chǔ)但重要的操作,傳統(tǒng)方法會使用一個臨時變量,但在某些情況下,我們可能需要不使用臨時變量來實(shí)現(xiàn)交換,本文將詳細(xì)介紹多種不使用臨時變量交換兩個變量值的方法,需要的朋友可以參考下
    2025-04-04
  • Java常量池知識點(diǎn)總結(jié)

    Java常量池知識點(diǎn)總結(jié)

    本篇文章給大家通過理論原理等方便徹底分析了Java常量池的相關(guān)知識,有興趣的朋友閱讀學(xué)習(xí)下吧。
    2017-12-12

最新評論