自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

還在用if(obj!=null)做非空判斷?帶你快速上手Optional實戰(zhàn)性理解!

開發(fā) 后端
今天就要盡可能的利用Java8的新特性 Optional來盡量簡化代碼同時高效處理NPE(Null Pointer Exception 空指針異常)

 [[326139]]

1.前言

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

2.認識Optional并使用

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

常規(guī)判斷: 

  1. //對象 人  
  2. //屬性有 name,age  
  3. Person person=new Person();  
  4. if (null==person){  
  5.    return "person為null";  
  6.  
  7. return person; 

使用Optional: 

  1. //對象 人  
  2. //屬性有 name,age  
  3. Person person=new Person();  
  4. return Optional.ofNullable(person).orElse("person為null"); 

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

  1. public class Person {  
  2.     private String name;  
  3.     private Integer age;  
  4.     public Person(String name, Integer age) {  
  5.         this.name = name;  
  6.         this.age = age;  
  7.     }  
  8.     public Person() {  
  9.     }  
  10.     public String getName() {  
  11.         return name;  
  12.     }  
  13.     public void setName(String name) {  
  14.         this.name = name;  
  15.     } 
  16.     public Integer getAge() {  
  17.         return age;  
  18.     }  
  19.     public void setAge(Integer age) {  
  20.         this.age = age;  
  21.     }  

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

2.1 Optional對象創(chuàng)建

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

  1. public final class Optional<T> {  
  2.    private static final Optional<?> EMPTY = new Optional<>();  
  3.    private final T value;  
  4.    //我們可以看到兩個構造方格都是private 私有的  
  5.    //說明 我們沒辦法在外面去new出來Optional對象  
  6.    private Optional() {  
  7.         this.value = null 
  8.     }  
  9.    private Optional(T value) {  
  10.         this.value = Objects.requireNonNull(value);  
  11.     }  
  12.     //這個靜態(tài)方法大致 是創(chuàng)建出一個包裝值為空的一個對象因為沒有任何參數(shù)賦值  
  13.    public static<T> Optional<T> empty() {  
  14.         @SuppressWarnings("unchecked")  
  15.         Optional<T> t = (Optional<T>) EMPTY;  
  16.         return t;  
  17.     }  
  18.     //這個靜態(tài)方法大致 是創(chuàng)建出一個包裝值非空的一個對象 因為做了賦值  
  19.    public static <T> Optional<T> of(T value) {  
  20.         return new Optional<>(value);  
  21.     }  
  22.     //這個靜態(tài)方法大致是 如果參數(shù)value為空,則創(chuàng)建空對象,如果不為空,則創(chuàng)建有參對象  
  23.    public static <T> Optional<T> ofNullable(T value) {  
  24.         return value == null ? empty() : of(value);  
  25.     }  
  26.  } 

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

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

我們關于創(chuàng)建Optional對象的內部方法大致分析完畢 接下來也正式的進入Optional的學習與使用中。更多Java面試技術點,在Java知音公眾號內回復“面試題聚合”

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

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

  1. public T get() {  
  2.     if (value == null) {  
  3.         throw new NoSuchElementException("No value present");  
  4.     }  
  5.     return value; 
  6.  

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

  1. Person person=new Person();  
  2. person.setAge(2); 
  3.  
  4. Optional.ofNullable(person).get(); 

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

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

  1. public Boolean isPresent() {  
  2.     return value != null; 
  3.  

簡單的實例展示: 

  1. Person person=new Person();  
  2. person.setAge(2);  
  3. if (Optional.ofNullable(person).isPresent()){  
  4.     //寫不為空的邏輯  
  5.     System.out.println("不為空");  
  6. } else{  
  7.     //寫為空的邏輯  
  8.     System.out.println("為空");  

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

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

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

看實例: 

  1. Person person=new Person();  
  2. person.setAge(2);  
  3. Optional.ofNullable(person).ifPresent(p -> System.out.println("年齡"+p.getAge())); 

如果對象不為空,則會打印這個年齡,因為內部已經做了NPE(非空判斷),所以就不用擔心空指針異常了。擴展知識點:巧用Java8中的Stream,讓集合操作飛起來!

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

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

源碼: 

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

簡單實例: 

  1. Person person=new Person();  
  2. person.setAge(2);  
  3. Optional.ofNullable(person).filter(p -> p.getAge()>50); 

2.6 Optional.map()方法(對象進行二次包裝) 

  1. map()方法將對應Funcation函數(shù)式接口中的對象,進行二次運算,封裝成新的對象然后返回在Optional中 源碼:  
  2.  public<U> Optional<U> map(Function<? super T, ? extends U> mapper) {  
  3.         Objects.requireNonNull(mapper);  
  4.         //如果為空返回自己  
  5.         if (!isPresent())  
  6.             return empty();  
  7.         else {  
  8.         //否則返回用方法修飾過的Optional  
  9.             return Optional.ofNullable(mapper.apply(value));  
  10.         }  
  11.     } 

實例展示: 

  1. Person person1=new Person();  
  2. person.setAge(2);  
  3. String optName = Optional.ofNullable(person).map(p -> person.getName()).orElse("name為空"); 

2.7 Optional.flatMap()方法(Optional對象進行二次包裝)

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

  1. public<U> Optional<U> flatMap(Function<? super T, Optional<U>> mapper) {  
  2.     Objects.requireNonNull(mapper);  
  3.     if (!isPresent())  
  4.                 return empty(); else {  
  5.         return Objects.requireNonNull(mapper.apply(value));  
  6.     }  

實例: 

  1. Person person=new Person();  
  2. person.setAge(2); 
  3. Optional<Object> optName = Optional.ofNullable(person).map(p -> Optional.ofNullable(p.getName()).orElse("name為空")); 

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

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

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

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

這個與orElse很相似,入?yún)⒉灰粯樱雲(yún)镾upplier對象,為空返回傳入對象的.get()方法,如果非空則返回當前對象 源碼: 

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

實例: 

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

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

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

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

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

源碼: 

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

實例:這個就貼實戰(zhàn)源碼了 

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

2.11 相似方法進行對比分析

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

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

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

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

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

map()和orElseGet的異同點

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

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

3.實戰(zhàn)場景再現(xiàn)

場景1:

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

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

場景2:

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

  1. public interface LocationRepository extends JpaRepository<Location, String> {  
  2. Optional<Location> findLocationById(String id);  

然在是Service中 

  1. public TerminalVO findById(String id) {  
  2. //這個方法在dao層也是用了Optional包裝了  
  3.         Optional<Terminal> terminalOptional = terminalRepository.findById(id);  
  4.         //直接使用isPresent()判斷是否為空  
  5.         if (terminalOptional.isPresent()) {  
  6.         //使用get()方法獲取對象值  
  7.             Terminal terminal = terminalOptional.get();  
  8.             //在實戰(zhàn)中,我們已經免去了用set去賦值的繁瑣,直接用BeanCopy去賦值  
  9.             TerminalVO terminalVO = BeanCopyUtils.copyBean(terminal, TerminalVO.class);  
  10.             //調用dao層方法返回包裝后的對象  
  11.             Optional<Location> location = locationRepository.findLocationById(terminal.getLocationId());  
  12.             if (location.isPresent()) {  
  13.                 terminalVO.setFullName(location.get().getFullName());  
  14.             }  
  15.             return terminalVO;  
  16.         }  
  17.         //不要忘記拋出異常  
  18.         throw new ServiceException("該終端不存在");  
  19.     } 

4.Optional使用注意事項

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

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

舉一個最簡單的栗子:

例子1:

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

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

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

5.jdk1.9對Optional優(yōu)化

首先增加了三個方法:

or()、ifPresentOrElse() 和 stream()

or() 與orElse等方法相似,如果對象不為空返回對象,如果為空則返回or()方法中預設的值。

ifPresentOrElse() 方法有兩個參數(shù):一個 Consumer 和一個 Runnable。如果對象不為空,會執(zhí)行 Consumer 的動作,否則運行 Runnable。相比ifPresent()多了OrElse判斷。

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

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

 

責任編輯:龐桂玉 來源: Java知音
相關推薦

2025-01-17 07:00:00

2022-02-24 07:56:42

開發(fā)Viteesbuild

2020-11-27 09:40:53

Rollup前端代碼

2021-12-06 17:44:56

MHAMySQL高可用

2022-05-05 09:14:41

AlpineDocker鏡像開發(fā)

2024-10-11 16:34:22

2024-10-15 15:58:11

2024-11-08 08:37:25

2022-07-22 09:15:07

OpitonalJava代碼

2012-07-19 10:03:32

2024-08-12 08:28:35

2023-11-18 09:17:56

Optional代碼

2022-11-14 18:43:03

JSCSS節(jié)流

2024-11-12 16:28:34

2023-10-08 11:09:22

Optional空指針

2024-04-11 09:17:51

ArraysJava安全

2021-10-14 18:15:38

BeanUtils對象生成器

2011-07-26 13:58:17

LINQ

2024-06-03 00:00:06

高性能數(shù)據(jù)傳輸應用程序

2020-08-20 09:17:02

數(shù)據(jù)分析技術IT
點贊
收藏

51CTO技術棧公眾號