Java8中新判空方法之Optional類的使用詳解
認(rèn)識(shí)Optional
Opitonal類就是Java提供的為了解決大家平時(shí)判斷對(duì)象是否為空用,通常會(huì)用 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");
測(cè)試展示類Student 代碼(如果有朋友不明白可以看一下這個(gè)):
@Data @AllArgsConstructor @NoArgsConstructor public?class?Student?{ ????private?String?name; ????private?Integer?age; }
Optional對(duì)象創(chuàng)建
首先我們先打開Optional的內(nèi)部,去一探究竟 先把幾個(gè)創(chuàng)建Optional對(duì)象的方法提取出來
public?final?class?Optional<T>?{ ???private?static?final?Optional<?>?EMPTY?=?new?Optional<>(); ???private?final?T?value; ???//?我們可以看到兩個(gè)構(gòu)造方格都是private?私有的 ???//?說明?我們沒辦法在外面去new出來Optional對(duì)象 ???private?Optional()?{ ????????this.value?=?null; ????} ???private?Optional(T?value)?{ ????????this.value?=?Objects.requireNonNull(value); ????} ????//?這個(gè)靜態(tài)方法大致?是創(chuàng)建出一個(gè)包裝值為空的一個(gè)對(duì)象因?yàn)闆]有任何參數(shù)賦值 ???public?static<T>?Optional<T>?empty()?{ ????????@SuppressWarnings("unchecked") ????????Optional<T>?t?=?(Optional<T>)?EMPTY; ????????return?t; ????} ????//?這個(gè)靜態(tài)方法大致?是創(chuàng)建出一個(gè)包裝值非空的一個(gè)對(duì)象?因?yàn)樽隽速x值 ???public?static?<T>?Optional<T>?of(T?value)?{ ????????return?new?Optional<>(value); ????} ????//?這個(gè)靜態(tài)方法大致是?如果參數(shù)value為空,則創(chuàng)建空對(duì)象,如果不為空,則創(chuàng)建有參對(duì)象 ???public?static?<T>?Optional<T>?ofNullable(T?value)?{ ????????return?value?==?null???empty()?:?of(value); ????} ?}
再做一個(gè)簡單的實(shí)例展示 與上面對(duì)應(yīng)
//?1、創(chuàng)建一個(gè)包裝對(duì)象值為空的Optional對(duì)象 Optional<String>?optEmpty?=?Optional.empty(); //?2、創(chuàng)建包裝對(duì)象值非空的Optional對(duì)象 Optional<String>?optOf?=?Optional.of("optional"); //?3、創(chuàng)建包裝對(duì)象值允許為空也可以不為空的Optional對(duì)象 Optional<String>?optOfNullable1?=?Optional.ofNullable(null); Optional<String>?optOfNullable2?=?Optional.ofNullable("optional");
Optional.get()方法(返回對(duì)象的值)
get()方法是返回一個(gè)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()方法就是會(huì)返回一個(gè)boolean類型值,如果對(duì)象不為空則為真,如果為空則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ù))
這個(gè)意思是如果對(duì)象非空,則運(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()));
如果對(duì)象不為空,則會(huì)打印這個(gè)年齡,因?yàn)閮?nèi)部已經(jīng)做了NPE(非空判斷),所以就不用擔(dān)心空指針異常了
Optional.filter()方法(過濾對(duì)象)
filter()方法大致意思是,接受一個(gè)對(duì)象,然后對(duì)他進(jìn)行條件過濾,如果條件符合則返回Optional對(duì)象本身,如果不符合則返回空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對(duì)象進(jìn)行二次包裝)
map()方法將對(duì)應(yīng)Optional< Funcation >
函數(shù)式接口中的對(duì)象,進(jìn)行二次運(yùn)算,封裝成新的對(duì)象然后返回在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()方法(為空返回對(duì)象)
常用方法之一,這個(gè)方法意思是如果包裝對(duì)象為空的話,就執(zhí)行orElse方法里的value,如果非空,則返回寫入對(duì)象
源碼:
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對(duì)象)
這個(gè)與orElse很相似,入?yún)⒉灰粯?,入?yún)镾upplier對(duì)象,為空返回傳入對(duì)象的.get()方法,如果非空則返回當(dāng)前對(duì)象。
源碼:
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()方法,此時(shí)才會(huì)調(diào)用對(duì)象的構(gòu)造方法,即獲得到真正對(duì)象 Optional.ofNullable(student).orElseGet(sup.get());
Suppiler是一個(gè)接口,是類似Spring的懶加載,聲明之后并不會(huì)占用內(nèi)存,只有執(zhí)行了get()方法之后,才會(huì)調(diào)用構(gòu)造方法創(chuàng)建出對(duì)象 創(chuàng)建對(duì)象的語法的話就是Supplier supStudent= Student::new;
需要使用時(shí)supStudent.get()
即可
Optional.orElseThrow()方法(為空返回異常)
方法作用的話就是如果為空,就拋出你定義的異常,如果不為空返回當(dāng)前對(duì)象,在實(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í)例:
//簡單的一個(gè)查詢 Member?member?=?memberService.selectByPhone(request.getPhone()); Optional.ofNullable(member).orElseThrow(()?->?new?ServiceException("沒有查詢的相關(guān)數(shù)據(jù)"));
相似方法進(jìn)行對(duì)比分析
可能看到這,沒用用過的話會(huì)覺得orElse()
和orElseGet()
還有orElseThrow()
很相似,map()和flatMap()好相似,不用著急,都是從這一步過來的,我再給大家總結(jié)一下不同方法的異同點(diǎn) orElse()
和orElseGet()
和orElseThrow()
的異同點(diǎn)
方法效果類似,如果對(duì)象不為空,則返回對(duì)象,如果為空,則返回方法體中的對(duì)應(yīng)參數(shù),所以可以看出這三個(gè)方法體中參數(shù)是不一樣的 orElse(T 對(duì)象) orElseGet(Supplier < T >
對(duì)象) orElseThrow(異常)
map()和orElseGet的異同點(diǎn)
方法效果類似,對(duì)方法參數(shù)進(jìn)行二次包裝,并返回,入?yún)⒉煌?map(function函數(shù)) flatmap(Optional< function >
函數(shù))
具體要怎么用,要根據(jù)業(yè)務(wù)場(chǎng)景以及代碼規(guī)范來定義,下面可以簡單看一下我在實(shí)戰(zhàn)中怎用使用神奇的Optional。
實(shí)戰(zhàn)場(chǎng)景
場(chǎng)景1:在service層中 查詢一個(gè)對(duì)象,返回之后判斷是否為空并做處理
//查詢一個(gè)對(duì)象 Member?member?=?memberService.selectByIdNo(request.getCertificateNo()); //使用ofNullable加orElseThrow做判斷和操作 Optional.ofNullable(member).orElseThrow(()?->?new?ServiceException("沒有查詢的相關(guān)數(shù)據(jù)"));
場(chǎng)景2:我們可以在dao接口層中定義返回值時(shí)就加上Optional
public?interface?LocationRepository?extends?JpaRepository<Location,?String>?{ Optional<Location>?findLocationById(String?id); }
然在是Service中
public?TerminalVO?findById(String?id)?{ ????//這個(gè)方法在dao層也是用了Optional包裝了 ????Optional<Terminal>?terminalOptional?=?terminalRepository.findById(id); ????//直接使用isPresent()判斷是否為空 ????if?(terminalOptional.isPresent())?{ ????//使用get()方法獲取對(duì)象值 ????????Terminal?terminal?=?terminalOptional.get(); ????????//在實(shí)戰(zhàn)中,我們已經(jīng)免去了用set去賦值的繁瑣,直接用BeanCopy去賦值 ????????TerminalVO?terminalVO?=?BeanCopyUtils.copyBean(terminal,?TerminalVO.class); ????????//調(diào)用dao層方法返回包裝后的對(duì)象 ????????Optional<Location>?location?=?locationRepository.findLocationById(terminal.getLocationId()); ????????if?(location.isPresent())?{ ????????????terminalVO.setFullName(location.get().getFullName()); ????????} ????????return?terminalVO; ????} ????//不要忘記拋出異常 ????throw?new?ServiceException("該終端不存在"); }
實(shí)戰(zhàn)場(chǎng)景還有很多,包括return時(shí)可以判斷是否返回當(dāng)前值還是跳轉(zhuǎn)到另一個(gè)方法體中,其它的還有很多。
Optional使用注意事項(xiàng)
Optional真么好用,真的可以完全替代if判斷嗎?我想這肯定是大家使用完之后Optional之后可能會(huì)產(chǎn)生的想法,答案是否定的 舉一個(gè)最簡單的栗子:
例子:如果我只想判斷對(duì)象的某一個(gè)變量是否為空并且做出判斷呢?
Person?person=new?Person(); person.setName(""); persion.setAge(2); //普通判斷 if(StringUtils.isNotBlank(person.getName())){ ???//名稱不為空?qǐng)?zhí)行代碼塊 } //使用Optional做判斷 Optional.ofNullable(person).map(p?->?p.getName()).orElse("name為空");
我覺得這個(gè)例子就能很好的說明這個(gè)問題,只是一個(gè)很簡單判斷,如果用了Optional我們還需要考慮包裝值,考慮代碼書寫,考慮方法調(diào)用,雖然只有一行,但是可讀性并不好,如果別的程序員去讀,我覺得肯定沒有if看的明顯。
Jdk 9對(duì)Optional優(yōu)化
首先增加了三個(gè)方法: or()
、ifPresentOrElse()
和 stream()
。or() 與orElse等方法相似,如果對(duì)象不為空返回對(duì)象,如果為空則返回or()方法中預(yù)設(shè)的值。
ifPresentOrElse()
方法有兩個(gè)參數(shù):一個(gè) Consumer 和一個(gè) Runnable。如果對(duì)象不為空,會(huì)執(zhí)行 Consumer 的動(dòng)作,否則運(yùn)行 Runnable。相比ifPresent()
多了OrElse判斷。stream()
將Optional轉(zhuǎn)換成stream,如果有值就返回包含值的stream,如果沒值,就返回空的stream。
以上就是Java8中新判空方法之Optional類的使用詳解的詳細(xì)內(nèi)容,更多關(guān)于Java8 Optional類的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Java Mybatis框架多表操作與注解開發(fā)詳解分析
MyBatis 是一款優(yōu)秀的持久層框架,它支持自定義 SQL、存儲(chǔ)過程以及高級(jí)映射。MyBatis 免除了幾乎所有的 JDBC 代碼以及設(shè)置參數(shù)和獲取結(jié)果集的工作。MyBatis 可以通過簡單的 XML 或注解來配置和映射原始類型、接口和 Java POJO為數(shù)據(jù)庫中的記錄2021-10-10SpringBoot+POI實(shí)現(xiàn)給word添加水印功能
這篇文章主要介紹了SpringBoot+POI實(shí)現(xiàn)給word添加水印功能,文中通過代碼示例講解的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作有一定的幫助,需要的朋友可以參考下2024-06-06基于java集合中的一些易混淆的知識(shí)點(diǎn)(詳解)
下面小編就為大家?guī)硪黄趈ava集合中的一些易混淆的知識(shí)點(diǎn)(詳解)。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2016-09-09Object類wait及notify方法原理實(shí)例解析
這篇文章主要介紹了Object類wait及notify方法原理實(shí)例解析,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-08-08SpringMVC之AbstractAnnotationConfigDispatcherSer解讀
這篇文章主要介紹了SpringMVC之AbstractAnnotationConfigDispatcherSer,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-05-05實(shí)例講解JAVA設(shè)計(jì)模式之備忘錄模式
這篇文章主要介紹了JAVA設(shè)計(jì)模式之備忘錄模式的的相關(guān)資料,文中示例代碼非常詳細(xì),供大家參考和學(xué)習(xí),感興趣的朋友可以了解下2020-06-06SpringMVC自定義攔截器登錄檢測(cè)功能的實(shí)現(xiàn)代碼
這篇文章主要介紹了SpringMVC自定義攔截器登錄檢測(cè)功能的實(shí)現(xiàn),本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-08-08