都2020年了,還在用if(obj!=null)作非空判斷??帶你快速上手Optional實戰性理解!

輕鬆實戰性理解Optionaljava

1.前言

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

2.認識Optional並使用

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

常規判斷:
        //對象 人
        //屬性有 name,age
        Person person=new Person();
       if (null==person){
            return "person爲null";
        }
        return person;
複製代碼
使用Optional:
        //對象 人
        //屬性有 name,age
        Person person=new Person();
        return Optional.ofNullable(person).orElse("person爲null");
複製代碼

測試展現類Person代碼(若是有朋友不明白能夠看一下這個):函數

public class Person {
    private String name;
    private Integer age;

    public Person(String name, Integer age) {
        this.name = name;
        this.age = age;
    }

    public Person() {
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public Integer getAge() {
        return age;
    }

    public void setAge(Integer age) {
        this.age = age;
    }
}
複製代碼

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

2.1 Optional對象建立

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

public final class Optional<T> {
   private static final Optional<?> EMPTY = new Optional<>();
   private final T value;
   //咱們能夠看到兩個構造方格都是private 私有的
   //說明 咱們沒辦法在外面去new出來Optional對象
   private Optional() {
        this.value = null;
    }
   private Optional(T value) {
        this.value = Objects.requireNonNull(value);
    }
    //這個靜態方法大體 是建立出一個包裝值爲空的一個對象由於沒有任何參數賦值
   public static<T> Optional<T> empty() {
        @SuppressWarnings("unchecked")
        Optional<T> t = (Optional<T>) EMPTY;
        return t;
    }
    //這個靜態方法大體 是建立出一個包裝值非空的一個對象 由於作了賦值
   public static <T> Optional<T> of(T value) {
        return new Optional<>(value);
    }
    //這個靜態方法大體是 若是參數value爲空,則建立空對象,若是不爲空,則建立有參對象
   public static <T> Optional<T> ofNullable(T value) {
        return value == null ? empty() : of(value);
    }
 }
複製代碼

再作一個簡單的實例展現 與上面對應優化

// 一、建立一個包裝對象值爲空的Optional對象
        Optional<String> optEmpty = Optional.empty();
        // 二、建立包裝對象值非空的Optional對象
        Optional<String> optOf = Optional.of("optional");
        // 三、建立包裝對象值容許爲空也能夠不爲空的Optional對象
        Optional<String> optOfNullable1 = Optional.ofNullable(null);
        Optional<String> optOfNullable2 = Optional.ofNullable("optional");
複製代碼

咱們關於建立Optional對象的內部方法大體分析完畢 接下來也正式的進入Optional的學習與使用中ui

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

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

public T get() {
        if (value == null) {
            throw new NoSuchElementException("No value present");
        }
        return value;
    }
複製代碼

也就是若是value不爲空則作返回,若是爲空則拋出異常 "No value present" 簡單實例展現spa

Person person=new Person();
        person.setAge(2);
        Optional.ofNullable(person).get();
複製代碼

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

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

public boolean isPresent() {
        return value != null;
    }
複製代碼

簡單的實例展現:

Person person=new Person();
        person.setAge(2);
        if (Optional.ofNullable(person).isPresent()){
        //寫不爲空的邏輯
        System.out.println("不爲空");
        }else{
         //寫爲空的邏輯
         System.out.println("爲空");
        }
複製代碼

2.4 Optional.ifPresent()方法(判讀是否爲空並返回函數)

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

public void ifPresent(Consumer<? super T> consumer) {
        //若是value不爲空,則運行accept方法體
        if (value != null)
            consumer.accept(value);
    }
複製代碼

看實例:

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

若是對象不爲空,則會打印這個年齡,由於內部已經作了NPE(非空判斷),因此就不用擔憂空指針異常了

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

filter()方法大體意思是,接受一個對象,而後對他進行條件過濾,若是條件符合則返回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();
    }
複製代碼

簡單實例:

Person person=new Person();
        person.setAge(2);
        Optional.ofNullable(person).filter(p -> p.getAge()>50);
複製代碼

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

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

public<U> Optional<U> map(Function<? super T, ? extends U> mapper) {
        Objects.requireNonNull(mapper);
        //若是爲空返回本身
        if (!isPresent())
            return empty();
        else {
        //不然返回用方法修飾過的Optional
            return Optional.ofNullable(mapper.apply(value));
        }
    }
複製代碼

實例展現:

Person person1=new Person();
        person.setAge(2);
        String optName = Optional.ofNullable(person).map(p -> person.getName()).orElse("name爲空");
複製代碼

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

map()方法將對應Optional< Funcation >函數式接口中的對象,進行二次運算,封裝成新的對象而後返回在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));
        }
    }
複製代碼

實例:

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

複製代碼

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

經常使用方法之一,這個方法意思是若是包裝對象爲空的話,就執行orElse方法裏的value,若是非空,則返回寫入對象 源碼:

public T orElse(T other) {
    //若是非空,返回value,若是爲空,返回other
        return value != null ? value : other;
    }
複製代碼

實例:

Person person1=new Person();
        person.setAge(2);
        Optional.ofNullable(person).orElse(new Person("小明", 2));
複製代碼

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

這個與orElse很類似,入參不同,入參爲Supplier對象,爲空返回傳入對象的.get()方法,若是非空則返回當前對象 源碼:

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

實例:

Optional<Supplier<Person>> sup=Optional.ofNullable(Person::new);
        //調用get()方法,此時纔會調用對象的構造方法,即得到到真正對象
         Optional.ofNullable(person).orElseGet(sup.get());
複製代碼

說真的對於Supplier對象我也懵逼了一下,去網上簡單查閱才得知 Supplier也是建立對象的一種方式,簡單來講,Suppiler是一個接口,是相似Spring的懶加載,聲明以後並不會佔用內存,只有執行了get()方法以後,纔會調用構造方法建立出對象 建立對象的語法的話就是Supplier<Person> supPerson= Person::new; 須要使用時supPerson.get()便可

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

這個我我的在實戰中也常常用到這個方法,方法做用的話就是若是爲空,就拋出你定義的異常,若是不爲空返回當前對象,在實戰中全部異常確定是要處理好的,爲了代碼的可讀性 源碼:

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

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

//簡單的一個查詢
        Member member = memberService.selectByPhone(request.getPhone());
        Optional.ofNullable(member).orElseThrow(() -> new ServiceException("沒有查詢的相關數據"));
複製代碼

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層中 查詢一個對象,返回以後判斷是否爲空並作處理

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

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

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();
            //在實戰中,咱們已經免去了用set去賦值的繁瑣,直接用BeanCopy去賦值
            TerminalVO terminalVO = BeanCopyUtils.copyBean(terminal, TerminalVO.class);
            //調用dao層方法返回包裝後的對象
            Optional<Location> location = locationRepository.findLocationById(terminal.getLocationId());
            if (location.isPresent()) {
                terminalVO.setFullName(location.get().getFullName());
            }
            return terminalVO;
        }
        //不要忘記拋出異常
        throw new ServiceException("該終端不存在");
    }
複製代碼

實戰場景還有不少,包括return時能夠判斷是否返回當前值仍是跳轉到另外一個方法體中,什麼的還有不少,若是你們沒有經驗的小夥伴還想進行學習,能夠評論一下我會回覆你們

4.Optional使用注意事項

Optional真麼好用,真的能夠徹底替代if判斷嗎? 我想這確定是你們使用完以後Optional以後可能會產生的想法,答案是否認的 舉一個最簡單的栗子: 例子1: 若是我只想判斷對象的某一個變量是否爲空而且作出判斷呢?

Person person=new Person();
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的優化,我就不深刻去說了。

做者的話

互相尊重,互相進步,很感謝你們的無私精神。才能讓咱們中國IT愈來愈進步 也很是願意虛心聽取更多大佬的意見和建議,和你們一塊兒交流進步 我是大家的好朋友 樊亦凡

若是你們以爲還不錯,但願能夠給我一個贊,很是感謝! 很是想和你們交朋友,和你們一塊兒進步! 一個天天進步一點點的程序員

相關文章
相關標籤/搜索