吴忠躺衫网络科技有限公司

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

認識Optional并使用

Android編程精選 ? 來源:稀土掘金技術社區 ? 作者:樊亦凡 ? 2022-09-05 14:18 ? 次閱讀

1、前言

相信不少小伙伴已經被java的NPE(Null Pointer Exception)所謂的空指針異常搞的頭昏腦漲,有大佬說過“防止 NPE,是程序員的基本修養。”但是修養歸修養,也是我們程序員最頭疼的問題之一,那么我們今天就要盡可能的利用Java8的新特性 Optional來盡量簡化代碼同時高效處理NPE(Null Pointer Exception 空指針異常)


2、認識Optional并使用

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

常規判斷:

//對象人
//屬性有name,age
Personperson=newPerson();
if(null==person){
return"person為null";
}
returnperson;

使用Optional:

//對象人
//屬性有name,age
Personperson=newPerson();
returnOptional.ofNullable(person).orElse("person為null");

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

publicclassPerson{
privateStringname;
privateIntegerage;

publicPerson(Stringname,Integerage){
this.name=name;
this.age=age;
}

publicPerson(){
}

publicStringgetName(){
returnname;
}

publicvoidsetName(Stringname){
this.name=name;
}

publicIntegergetAge(){
returnage;
}

publicvoidsetAge(Integerage){
this.age=age;
}
}

下面,我們就高效的學習一下神奇的Optional類!

2.1 Optional對象創建

首先我們先打開Optional的內部,去一探究竟 先把幾個創建Optional對象的方法提取出來

publicfinalclassOptional<T>{
privatestaticfinalOptionalEMPTY=newOptional<>();
privatefinalTvalue;
//我們可以看到兩個構造方格都是private私有的
//說明我們沒辦法在外面去new出來Optional對象
privateOptional(){
this.value=null;
}
privateOptional(Tvalue){
this.value=Objects.requireNonNull(value);
}
//這個靜態方法大致是創建出一個包裝值為空的一個對象因為沒有任何參數賦值
publicstaticOptionalempty(){
@SuppressWarnings("unchecked")
Optionalt=(Optional)EMPTY;
returnt;
}
//這個靜態方法大致是創建出一個包裝值非空的一個對象因為做了賦值
publicstaticOptionalof(Tvalue){
returnnewOptional<>(value);
}
//這個靜態方法大致是如果參數value為空,則創建空對象,如果不為空,則創建有參對象
publicstaticOptionalofNullable(Tvalue){
returnvalue==null?empty():of(value);
}
}

再做一個簡單的實例展示 與上面對應

//1、創建一個包裝對象值為空的Optional對象
OptionaloptEmpty=Optional.empty();
//2、創建包裝對象值非空的Optional對象
OptionaloptOf=Optional.of("optional");
//3、創建包裝對象值允許為空也可以不為空的Optional對象
OptionaloptOfNullable1=Optional.ofNullable(null);
OptionaloptOfNullable2=Optional.ofNullable("optional");

我們關于創建Optional對象的內部方法大致分析完畢 接下來也正式的進入Optional的學習與使用中

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

get()方法是返回一個option的實例值 源碼:

publicTget(){
if(value==null){
thrownewNoSuchElementException("Novaluepresent");
}
returnvalue;
}

也就是如果value不為空則做返回,如果為空則拋出異常 "No value present" 簡單實例展示

Personperson=newPerson();
person.setAge(2);
Optional.ofNullable(person).get();
2.3 Optional.isPresent()方法(判讀是否為空)

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

publicbooleanisPresent(){
returnvalue!=null;
}

簡單的實例展示:

Personperson=newPerson();
person.setAge(2);
if(Optional.ofNullable(person).isPresent()){
//寫不為空的邏輯
System.out.println("不為空");
}else{
//寫為空的邏輯
System.out.println("為空");
}
2.4 Optional.ifPresent()方法(判讀是否為空并返回函數)

這個意思是如果對象非空,則運行函數體 源碼:

publicvoidifPresent(ConsumersuperT>consumer){
//如果value不為空,則運行accept方法體
if(value!=null)
consumer.accept(value);
}

看實例:

Personperson=newPerson();
person.setAge(2);
Optional.ofNullable(person).ifPresent(p->System.out.println("年齡"+p.getAge()));

如果對象不為空,則會打印這個年齡,因為內部已經做了NPE(非空判斷),所以就不用擔心空指針異常了

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

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

源碼:

publicOptionalfilter(PredicatesuperT>predicate){
Objects.requireNonNull(predicate);
//如果為空直接返回this
if(!isPresent())
returnthis;
else
//判斷返回本身還是空Optional
returnpredicate.test(value)?this:empty();
}

簡單實例:

Personperson=newPerson();
person.setAge(2);
Optional.ofNullable(person).filter(p->p.getAge()>50);
2.6 Optional.map()方法(對象進行二次包裝)

map()方法將對應Funcation函數式接口中的對象,進行二次運算,封裝成新的對象然后返回在Optional中 源碼:

publicOptionalmap(FunctionsuperT,?extendsU>mapper){
Objects.requireNonNull(mapper);
//如果為空返回自己
if(!isPresent())
returnempty();
else{
//否則返回用方法修飾過的Optional
returnOptional.ofNullable(mapper.apply(value));
}
}

實例展示:

Personperson1=newPerson();
person.setAge(2);
StringoptName=Optional.ofNullable(person).map(p->person.getName()).orElse("name為空");
2.7 Optional.flatMap()方法(Optional對象進行二次包裝)

map()方法將對應Optional< Funcation >函數式接口中的對象,進行二次運算,封裝成新的對象然后返回在Optional中 源碼:

publicOptionalflatMap(FunctionsuperT,Optional>mapper){
Objects.requireNonNull(mapper);
if(!isPresent())
returnempty();
else{
returnObjects.requireNonNull(mapper.apply(value));
}
}

實例:

Personperson=newPerson();
person.setAge(2);
OptionaloptName=Optional.ofNullable(person).map(p->Optional.ofNullable(p.getName()).orElse("name為空"));

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

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

publicTorElse(Tother){
//如果非空,返回value,如果為空,返回other
returnvalue!=null?value:other;
}

實例:

Personperson1=newPerson();
person.setAge(2);
Optional.ofNullable(person).orElse(newPerson("小明",2));
2.9 Optional.orElseGet()方法(為空返回Supplier對象)

這個與orElse很相似,入參不一樣,入參為Supplier對象,為空返回傳入對象的.get()方法,如果非空則返回當前對象 源碼:

publicTorElseGet(Supplierother){
returnvalue!=null?value:other.get();
}

實例:

Optional>sup=Optional.ofNullable(Person::new);
//調用get()方法,此時才會調用對象的構造方法,即獲得到真正對象
Optional.ofNullable(person).orElseGet(sup.get());

說真的對于Supplier對象我也懵逼了一下,去網上簡單查閱才得知 Supplier也是創建對象的一種方式,簡單來說,Suppiler是一個接口,是類似Spring的懶加載,聲明之后并不會占用內存,只有執行了get()方法之后,才會調用構造方法創建出對象創建對象的語法的話就是Supplier supPerson= Person::new;

需要使用時supPerson.get()即可

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

這個我個人在實戰中也經常用到這個方法,方法作用的話就是如果為空,就拋出你定義的異常,如果不為空返回當前對象,在實戰中所有異常肯定是要處理好的,為了代碼的可讀性

源碼:

publicTorElseThrow(SupplierexceptionSupplier)throwsX{
if(value!=null){
returnvalue;
}else{
throwexceptionSupplier.get();
}
}

實例:這個就貼實戰源碼了

//簡單的一個查詢
Membermember=memberService.selectByPhone(request.getPhone());
Optional.ofNullable(member).orElseThrow(()->newServiceException("沒有查詢的相關數據"));
2.11 相似方法進行對比分析

可能小伙伴看到這,沒用用過的話會覺得orElse()orElseGet()還有orElseThrow()很相似,map()flatMap()好相似

哈哈哈不用著急,都是從這一步過來的,我再給大家總結一下不同方法的異同點

orElse()和orElseGet()和orElseThrow()的異同點

方法效果類似,如果對象不為空,則返回對象,如果為空,則返回方法體中的對應參數,所以可以看出這三個方法體中參數是不一樣的

  • orElse(T 對象)
  • orElseGet(Supplier < T >對象)
  • orElseThrow(異常)

map()和orElseGet的異同點

方法效果類似,對方法參數進行二次包裝,并返回,入參不同

  • map(function函數)
  • flatmap(Optional< function >函數)

具體要怎么用,要根據業務場景以及代碼規范來定義,下面可以簡單看一下我在實戰中怎用使用神奇的Optional

3、實戰場景再現

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

//查詢一個對象
Membermember=memberService.selectByIdNo(request.getCertificateNo());
//使用ofNullable加orElseThrow做判斷和操作
Optional.ofNullable(member).orElseThrow(()->newServiceException("沒有查詢的相關數據"));

場景2:我們可以在dao接口層中定義返回值時就加上Optional 例如:我使用的是jpa,其他也同理

publicinterfaceLocationRepositoryextendsJpaRepository<Location,String>{
OptionalfindLocationById(Stringid);
}

然后在是Service中

publicTerminalVOfindById(Stringid){
//這個方法在dao層也是用了Optional包裝了
OptionalterminalOptional=terminalRepository.findById(id);
//直接使用isPresent()判斷是否為空
if(terminalOptional.isPresent()){
//使用get()方法獲取對象值
Terminalterminal=terminalOptional.get();
//在實戰中,我們已經免去了用set去賦值的繁瑣,直接用BeanCopy去賦值
TerminalVOterminalVO=BeanCopyUtils.copyBean(terminal,TerminalVO.class);
//調用dao層方法返回包裝后的對象
Optionallocation=locationRepository.findLocationById(terminal.getLocationId());
if(location.isPresent()){
terminalVO.setFullName(location.get().getFullName());
}
returnterminalVO;
}
//不要忘記拋出異常
thrownewServiceException("該終端不存在");
}

實戰場景還有很多,包括return時可以判斷是否返回當前值還是跳轉到另一個方法體中,什么的還有很多,如果大家沒有經驗的小伙伴還想進行學習,可以評論一下我會回復大家

4、Optional使用注意事項

Optional真么好用,真的可以完全替代if判斷嗎?

我想這肯定是大家使用完之后Optional之后可能會產生的想法,答案是否定的

舉一個最簡單的栗子:

例子1:

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

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

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

5、jdk1.9對Optional優化


首先增加了三個方法:

  • or()ifPresentOrElse()stream()
  • or()orElse等方法相似,如果對象不為空返回對象,如果為空則返回or()方法中預設的值。
  • ifPresentOrElse() 方法有兩個參數:一個 Consumer 和一個 Runnable。如果對象不為空,會執行 Consumer 的動作,否則運行 Runnable。相比ifPresent()多了OrElse判斷。

stream() 將Optional轉換成stream,如果有值就返回包含值的stream,如果沒值,就返回空的stream。

因為這個jdk1.9的Optional具體我沒有測試,同時也發現有蠻好的文章已經也能讓大家明白jdk1.9的option的優化,我就不深入去說了

審核編輯:湯梓紅

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • JAVA
    +關注

    關注

    19

    文章

    2974

    瀏覽量

    105136
  • 函數
    +關注

    關注

    3

    文章

    4346

    瀏覽量

    62968

原文標題:不要再用 if (obj != null) 判空了!!!

文章出處:【微信號:AndroidPush,微信公眾號:Android編程精選】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    JDK8新增的Optional類的常用方法

    一、摘要 NullPointerException,中文名: 空指針異常 ,也簡稱 NPE,是軟件系統中最常見的錯誤異常之一。 很久以前 Google Guava 項目引入了 Optional 作為
    的頭像 發表于 09-30 15:13 ?638次閱讀

    SPI時序認識

    實時時鐘DS1302一、SPI時序認識一、SPI時序認識extern只用于外部變量的聲明,而不是定義。我們需要在第一次用到該變量的.c文件中聲明定義該變量,其他的.c文件如果需要再次使用該變量我們
    發表于 11-22 07:10

    認識燃料電池

    認識燃料電池發展起源􀂊 1839年英國的William Grove發明了燃料電池􀂊 60年代英國康橋大學的Francis Bacon用高壓氫氧制成了具有實用性質的燃料電池運用
    發表于 11-03 10:43 ?14次下載

    認識氧化亞鈷

    認識氧化亞鈷  
    發表于 10-21 14:54 ?1283次閱讀

    認識錳酸鋰產品

    認識錳酸鋰產品
    發表于 10-29 11:34 ?625次閱讀

    JDK8 Optional類新特性

    Optional不是對null關鍵字的一種替代,而是對于null判定提供了一種更加優雅的實現。 NullPointException可以說是所有java程序員都遇到過的一個異常,雖然java從
    發表于 11-28 14:33 ?1802次閱讀

    電路的構成及對電源的認識

    電源的基本認識
    的頭像 發表于 08-14 00:46 ?3507次閱讀

    JAVA8提供了Optional類來優化這種寫法

    這種寫法是比較丑陋的,為了避免上述丑陋的寫法,讓丑陋的設計變得優雅。JAVA8提供了Optional類來優化這種寫法,接下來的正文部分進行詳細說明
    的頭像 發表于 04-24 15:18 ?1124次閱讀

    E10A-USB Optional 38-Pin User System Interface Cable HS0005ECK01H 用戶手冊

    E10A-USB Optional 38-Pin User System Interface Cable HS0005ECK01H 用戶手冊
    發表于 05-06 18:58 ?0次下載
    E10A-USB <b class='flag-5'>Optional</b> 38-Pin User System Interface Cable HS0005ECK01H 用戶手冊

    E10A-USB Optional 38-Pin User System Interface Cable HS0005ECK01H 用戶手冊

    E10A-USB Optional 38-Pin User System Interface Cable HS0005ECK01H 用戶手冊
    發表于 06-26 20:44 ?0次下載
    E10A-USB <b class='flag-5'>Optional</b> 38-Pin User System Interface Cable HS0005ECK01H 用戶手冊

    認識CAN總線錯誤 CAN總線錯誤分析與解決

    認識CAN 總線錯誤的第一步就是了解認識CAN 總線協議和它的具體功能,這樣才能更容易地理解CAN 總線是如何去發現解決錯誤的。
    發表于 08-14 15:18 ?6667次閱讀
    <b class='flag-5'>認識</b>CAN總線錯誤 CAN總線錯誤分析與解決

    認識靜電?靜電是什么?

    認識靜電?靜電是什么?|深圳比創達EMC
    的頭像 發表于 10-08 10:48 ?2207次閱讀

    簡單認識無源器件

    簡單認識無源器件
    的頭像 發表于 01-12 09:56 ?640次閱讀
    簡單<b class='flag-5'>認識</b>無源器件

    如何認識伺服電機

    電子發燒友網站提供《如何認識伺服電機.zip》資料免費下載
    發表于 03-22 17:43 ?0次下載

    反射內存卡的基本認識

    教你認識反射內存卡
    的頭像 發表于 11-14 14:28 ?575次閱讀
    反射內存卡的基本<b class='flag-5'>認識</b>
    百家乐稳赢投资法| 百家乐官网现场投注平台| 哪个百家乐最好| 大发888游戏下载投注| 百家乐官网娱乐软件| 威尼斯人娱乐城返佣| 百家乐官网推二八杠| 威尼斯人娱乐代理注测| 百家乐官网博娱乐赌百家乐官网的玩法技巧和规则 | 丽江市| 专业百家乐筹码| 元氏县| 百家乐一邱大师打法| 百家乐官网大转轮| 顶级赌场官方下载| 网络百家乐可靠吗| 百家乐官网游戏排行榜| 破战百家乐的玩法技巧和规则| 个人百家乐官网策略| 星空棋牌下载| 菲律宾百家乐试玩| 缅甸百家乐官网网上投注| 百家乐设备电子路| 澳门百家乐官网图形| 体球网足球比分| 澳门百家乐规则视频| 玩百家乐官网输澳门百家乐官网现场 | 太阳城娱乐城| 百家乐闲和庄| 海立方百家乐官网海立方| 大发888官方 df888gfxzylc8| 24山水口吉凶图| 真人百家乐官网海立方| 太阳城金旭园| 利记百家乐现金网| 乐九百家乐官网现金网| 大发888新址| 百家乐只打一种牌型| 百家乐官网西园二手房| 皇冠网都市小说| 百家乐博娱乐网赌百家乐的玩法技巧和规则 |