激情久久久_欧美视频区_成人av免费_不卡视频一二三区_欧美精品在欧美一区二区少妇_欧美一区二区三区的

服務器之家:專注于服務器技術及軟件下載分享
分類導航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術|正則表達式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務器之家 - 編程語言 - Java教程 - 淺談Java8 判空新寫法

淺談Java8 判空新寫法

2022-01-21 00:49Mr泰迪 Java教程

在開發過程中很多時候會遇到判空校驗,如果不做判空校驗則會產生NullPointerException異常,本文就來介紹一下Java8 判空新寫法,感興趣的可以了解一下

引言

在開發過程中很多時候會遇到判空校驗,如果不做判空校驗則會產生NullPointerException異常。如下代碼直接使用可能會有問題。

user.getAddress().getProvince();

當user為null時,會產生空指針異常,常規解決辦法如下:

if(user!=null){
  Address address = user.getAddress();
  if(address!=null){
      String province = address.getProvince();
  }
}

這種寫法是比較丑陋的,為了避免上述丑陋的寫法,讓丑陋的設計變得優雅。Java8提供了Optional類來優化這種寫法。

 

API介紹

先介紹一下API,與其他文章不同的是,本文采取類比的方式來講,同時結合源碼。而不像其他文章一樣,一個個API羅列出來,讓人找不到重點。

1、Optional(),empty(),of(),ofNullable()

這四個函數之間具有相關性,因此放在一組進行記憶。

先說明一下,Optional(T value),即構造函數,它是private權限的,不能由外部調用的。其余三個函數是public權限,供我們所調用。那么,Optional的本質,就是內部儲存了一個真實的值,在構造的時候,就直接判斷其值是否為空。好吧,這么說還是比較抽象。直接上Optional(T value)構造函數的源碼,如下圖所示

淺談Java8 判空新寫法

那么,of(T value)的源碼如下

public static <T> Optional<T> of(T value) {
  return new Optional<>(value);
}

也就是說of(T value)函數內部調用了構造函數。根據構造函數的源碼我們可以得出兩個結論:

  • 通過of(T value)函數所構造出的Optional對象,當Value值為空時,依然會報NullPointerException
  • 通過of(T value)函數所構造出的Optional對象,當Value值不為空時,能正常構造Optional對象。

除此之外呢,Optional類內部還維護一個value為null的對象,大概就是長下面這樣的

public final class Optional<T> {
  //省略....
  private static final Optional<?> EMPTY = new Optional<>();
  private Optional() {
      this.value = null;
  }
  //省略...
  public static<T> Optional<T> empty() {
      @SuppressWarnings("unchecked")
      Optional<T> t = (Optional<T>) EMPTY;
      return t;
  }
}

那么,empty()的作用就是返回EMPTY對象。好了鋪墊了這么多,可以說ofNullable(T value)的作用了,上源碼

public static <T> Optional<T> ofNullable(T value) {
  return value == null ? empty() : of(value);
}

好吧,大家應該都看得懂什么意思了。相比較of(T value)的區別就是:

  • 當value值為null時,of(T value)會報NullPointerException異常;
  • ofNullable(T value)不會throw Exception,ofNullable(T value)直接返回一個EMPTY對象。

那是不是意味著,我們在項目中只用ofNullable函數而不用of函數呢?
不是的,一個東西存在那么自然有存在的價值。當在運行過程中,不想隱藏NullPointerException。而是要立即報告,這種情況下就用Of函數。但是不得不承認,這樣的場景真的很少。博主也僅在寫junit測試用例中用到過此函數。

2、orElse(),orElseGet()和orElseThrow()

這三個函數放一組進行記憶,都是在構造函數傳入的value值為null時,進行調用的。orElse和orElseGet的用法如下所示,相當于value值為null時,給予一個默認值:

@Test
public void test() {
  User user = null;
  user = Optional.ofNullable(user).orElse(createUser());
  user = Optional.ofNullable(user).orElseGet(() -> createUser());

}
public User createUser(){
  User user = new User();
  user.setName("zhangsan");
  return user;
}

這兩個函數的區別:當user值不為null時,orElse函數依然會執行createUser()方法,而orElseGet函數并不會執行createUser()方法,大家可自行測試。

至于orElseThrow,就是value值為null時,直接拋一個異常出去,用法如下所示:

User user = null;
Optional.ofNullable(user).orElseThrow(()->new Exception("用戶不存在"));

3、map()和flatMap()

這兩個函數放在一組記憶,這兩個函數做的是轉換值的操作。

直接上源碼

public final class Optional<T> {
  //省略....
   public<U> Optional<U> map(Function<? super T, ? extends U> mapper) {
      Objects.requireNonNull(mapper);
      if (!isPresent())
          return empty();
      else {
          return Optional.ofNullable(mapper.apply(value));
      }
  }
  //省略...
   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));
      }
  }
}

這兩個函數,在函數體上沒什么區別。唯一區別的就是入參,map函數所接受的入參類型為Function<? super T, ? extends U>,而flapMap的入參類型為Function<? super T, Optional>。

在具體用法上,對于map而言:

如果User結構是下面這樣的

public class User {
  private String name;
  public String getName() {
      return name;
  }
}

這時候取name的寫法如下所示

String city = Optional.ofNullable(user).map(u-> u.getName()).get();

對于flatMap而言:
如果User結構是下面這樣的

public class User {
  private String name;
  public Optional<String> getName() {
      return Optional.ofNullable(name);
  }
}

這時候取name的寫法如下所示

String city = Optional.ofNullable(user).flatMap(u-> u.getName()).get();

4、isPresent()和ifPresent(Consumer<? super T> consumer)

這兩個函數放在一起記憶,isPresent即判斷value值是否為空,而ifPresent就是在value值不為空時,做一些操作。這兩個函數的源碼如下

public final class Optional<T> {
  //省略....
  public boolean isPresent() {
      return value != null;
  }
  //省略...
  public void ifPresent(Consumer<? super T> consumer) {
      if (value != null)
          consumer.accept(value);
  }
}

5、filter(Predicate<? super T> predicate)

源碼:

public final class Optional<T> {
  //省略....
 Objects.requireNonNull(predicate);
      if (!isPresent())
          return this;
      else
          return predicate.test(value) ? this : empty();
}

filter 方法接受一個 Predicate 來對 Optional 中包含的值進行過濾,如果包含的值滿足條件,那么還是返回這個 Optional;否則返回 Optional.empty。

用法如下

Optional<User> user1 = Optional.ofNullable(user).filter(u -> u.getName().length()<6);

如上所示,如果user的name的長度是小于6的,則返回。如果是大于6的,則返回一個EMPTY對象。

 

實戰

例一

在函數方法中,以前寫法

public String getCity(User user)  throws Exception{
      if(user!=null){
          if(user.getAddress()!=null){
              Address address = user.getAddress();
              if(address.getCity()!=null){
                  return address.getCity();
              }
          }
      }
      throw new Excpetion("取值錯誤");
  }

JAVA8寫法

public String getCity(User user) throws Exception{
  return Optional.ofNullable(user)
                 .map(u-> u.getAddress())
                 .map(a->a.getCity())
                 .orElseThrow(()->new Exception("取指錯誤"));
}

例二

比如,在主程序中,以前寫法

if(user!=null){
  dosomething(user);
}

JAVA8寫法

Optional.ofNullable(user)
  .ifPresent(u->{
      dosomething(u);
});

例三

以前寫法

public User getUser(User user) throws Exception{
  if(user!=null){
      String name = user.getName();
      if("zhangsan".equals(name)){
          return user;
      }
  }else{
      user = new User();
      user.setName("zhangsan");
      return user;
  }
}

java8寫法

public User getUser(User user) {
  return Optional.ofNullable(user)
                 .filter(u->"zhangsan".equals(u.getName()))
                 .orElseGet(()-> {
                      User user1 = new User();
                      user1.setName("zhangsan");
                      return user1;
                 });
}

其他的例子,不一一列舉了。不過采用這種鏈式編程,雖然代碼優雅了。但是,邏輯性沒那么明顯,可讀性有所降低,大家項目中看情況酌情使用。

到此這篇關于淺談Java8 判空新寫法的文章就介紹到這了,更多相關Java8 判空內容請搜索服務器之家以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持服務器之家!

原文鏈接:https://juejin.cn/post/7012126634971152391

延伸 · 閱讀

精彩推薦
  • Java教程xml與Java對象的轉換詳解

    xml與Java對象的轉換詳解

    這篇文章主要介紹了xml與Java對象的轉換詳解的相關資料,需要的朋友可以參考下...

    Java教程網2942020-09-17
  • Java教程升級IDEA后Lombok不能使用的解決方法

    升級IDEA后Lombok不能使用的解決方法

    最近看到提示IDEA提示升級,尋思已經有好久沒有升過級了。升級完畢重啟之后,突然發現好多錯誤,本文就來介紹一下如何解決,感興趣的可以了解一下...

    程序猿DD9332021-10-08
  • Java教程Java8中Stream使用的一個注意事項

    Java8中Stream使用的一個注意事項

    最近在工作中發現了對于集合操作轉換的神器,java8新特性 stream,但在使用中遇到了一個非常重要的注意點,所以這篇文章主要給大家介紹了關于Java8中S...

    阿杜7482021-02-04
  • Java教程Java使用SAX解析xml的示例

    Java使用SAX解析xml的示例

    這篇文章主要介紹了Java使用SAX解析xml的示例,幫助大家更好的理解和學習使用Java,感興趣的朋友可以了解下...

    大行者10067412021-08-30
  • Java教程小米推送Java代碼

    小米推送Java代碼

    今天小編就為大家分享一篇關于小米推送Java代碼,小編覺得內容挺不錯的,現在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧...

    富貴穩中求8032021-07-12
  • Java教程Java實現搶紅包功能

    Java實現搶紅包功能

    這篇文章主要為大家詳細介紹了Java實現搶紅包功能,采用多線程模擬多人同時搶紅包,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙...

    littleschemer13532021-05-16
  • Java教程20個非常實用的Java程序代碼片段

    20個非常實用的Java程序代碼片段

    這篇文章主要為大家分享了20個非常實用的Java程序片段,對java開發項目有所幫助,感興趣的小伙伴們可以參考一下 ...

    lijiao5352020-04-06
  • Java教程Java BufferWriter寫文件寫不進去或缺失數據的解決

    Java BufferWriter寫文件寫不進去或缺失數據的解決

    這篇文章主要介紹了Java BufferWriter寫文件寫不進去或缺失數據的解決方案,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望...

    spcoder14552021-10-18
主站蜘蛛池模板: 亚洲国产精品久久久久久久久 | 欧美a级大胆视频 | 欧美中文在线 | 国内精品伊人久久久久网站 | 免费久久久 | 国产小视频一区 | av日韩在线免费观看 | 91短视频在线免费观看 | 国产精品久久久久国产精品三级 | 国产一区二区三区四区波多野结衣 | 在线成人影视 | 久久久一区二区三区精品 | 国产区二区 | 久久久成人免费视频 | 久久久久一区二区三区四区五区 | 亚洲天堂岛国片 | 日本精品视频一区二区三区四区 | 日本在线一区二区 | 激情亚洲一区二区 | 99久在线视频 | 狠狠操电影 | 国产精品99久久久久久宅女 | 91短视频在线免费观看 | 国产免费永久在线观看 | 日韩视频在线一区二区三区 | 99riav视频一区二区 | 少妇一级淫片免费放播放 | 在线成人免费网站 | 性色吧 | 国产精品久久久久久久久久久久午夜 | 99麻豆久久久国产精品免费 | 亚洲成人激情在线 | 久草在线资源福利站 | 欧美成年性h版影视中文字幕 | 久久网站免费 | 毛片午夜 | 27xxoo无遮挡动态视频 | 九九黄色 | 久久精品视频一区二区三区 | 亚洲午夜激情网 | 全黄性性激高免费视频 |