真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網站制作重慶分公司

利用Java8Optional避免空指針異常的示例

小編給大家分享一下利用Java8 Optional避免空指針異常的示例,希望大家閱讀完這篇文章之后都有所收獲,下面讓我們一起去探討吧!

創(chuàng)新互聯(lián)成都網站建設按需求定制網站,是成都網站營銷推廣公司,為崗亭提供網站建設服務,有成熟的網站定制合作流程,提供網站定制設計服務:原型圖制作、網站創(chuàng)意設計、前端HTML5制作、后臺程序開發(fā)等。成都網站營銷推廣熱線:18982081108

前言

空指針是我們最常見也最討厭的異常,為了防止空指針異常,你不得在代碼里寫大量的非空判斷。

Java 8引入了一個新的Optional類。用于避免空指針的出現(xiàn),也無需在寫大量的if(obj!=null)這樣的判斷了,前提是你得將數據用Optional裝著,它就是一個包裹著對象的容器。

都說沒有遇到過空指針異常的程序員不是Java程序員,null確實引發(fā)過很多問題。Java 8中引入了一個叫做java.util.Optional的新類可以避免null引起的諸多問題。

我們看看一個null引用能導致哪些危害。首先創(chuàng)建一個類Computer,結構如下圖所示:

利用Java8 Optional避免空指針異常的示例

當我們調用如下代碼會怎樣?

String version = computer.getSoundcard().getUSB().getVersion();

上述代碼看似是沒有問題的,但是很多計算機(比如,樹莓派)其實是沒有聲卡的,那么調用getSoundcard()方法可定會拋出空指針異常了。

一個常規(guī)的但是不好的的方法是返回一個null引用來表示計算機沒有聲卡,但是這就意味著會對一個空引調用getUSB()方法,顯然會在程序運行過程中拋出控制異常,從而導致程序停止運行。想想一下,當你的程序在客戶端電腦上運行時,突然出現(xiàn)這種錯是多尷尬的一件事?

偉大計算機科學Tony Hoare曾經寫到:"我認為null引用從1965年被創(chuàng)造出來導致了十億美元的損失。當初使用null引用對我最大的誘惑就是它實現(xiàn)起來方便。"

那么該怎么避免在程序運行時會出現(xiàn)空指針異常呢?你需要保持警惕,并且不斷檢查可能出現(xiàn)空指針的情況,就像下面這樣:

String version = "UNKNOWN";
if(computer != null)
 {
 Soundcard soundcard = computer.getSoundcard();
 if(soundcard != null){
  USB usb = soundcard.getUSB();
  if(usb != null){
   version = usb.getVersion();
  }
  }
 }

然而,你可以看到上述代碼有太多的null檢查,整個代碼結構變得非常丑陋。但是我們又不得不通過這樣的判斷來確保系統(tǒng)運行時不會出現(xiàn)空指針。如果在我們的業(yè)務代碼中出現(xiàn)大量的這種空引用判斷簡直讓人惱火,也導致我們代碼的可讀性會很差。

如果你忘記檢查要給值是否為空,null引用也是存在很大的潛在問題。這篇文章我將證明使用null引用作為值不存在的表示是不好的方法。我們需要一個更好的表示值不存在的模型,而不是再使用null引用。

Java 8引入了一個新類叫做java.util.Optional ,這個類的設計的靈感來源于Haskell語言和Scala語言。這個類可以包含了一個任意值,像下面圖和代碼表示的那樣。你可以把Optional看做是一個有可能包含了值的值,如果Optional不包含值那么它就是空的,下圖那樣。

利用Java8 Optional避免空指針異常的示例

public class Computer {
 private Optional soundcard;
 public Optional getSoundcard() { ... }
 ...
}

public class Soundcard {
 private Optional usb;
 public Optional getUSB() { ... }

}

public class USB{
 public String getVersion(){ ... }
}

上述代碼展現(xiàn)了一臺計算機有可能包換一個聲卡(聲卡是有可能存在也有可能不存在)。聲卡也是有可能包含一個USB端口的。這是一種改善方法,該模型可以更加清晰的反映一個被給定的值是可以不存在的。

但是該怎么處理Optional這個對象呢?畢竟,你想要獲取的是USB的端口號。很簡單,Optional類包含了一些方法來處理值是否存在的狀況。和null引用相比Optional類迫使你在你要做值是否相關處理,從而避免了空指針異常。

需要說明的是Optional類并不是要取代null引用。相反地,是為了讓設計的API更容易被理解,當你看到一個函數的簽名時,你就可以判斷要傳遞給這個函數的值是不是有可能不存在。這就促使你要打開Optional類來處理確實值的狀況了。

采用Optional模式

啰嗦了這么多,來看一些代碼吧!我們先看一下怎么使用Optional改寫傳統(tǒng)的null引用檢測后是什么樣子。在這邊文章的末尾你將會明白怎么使用Optional。

String name = computer.flatMap(Computer::getSoundcard)
       .flatMap(Soundcard::getUSB)
       .map(USB::getVersion)
       .orElse("UNKNOWN");

創(chuàng)建Optional對象

可以創(chuàng)建一個空的Optional對象:

Optional sc = Optional.empty();

接下來是創(chuàng)建一個包含非null值的Optional:

SoundCard soundcard = new Soundcard();
Optional sc = Optional.of(soundcard);

如果聲卡null,空指針異常會立即被拋出(這比在獲取聲卡屬性時才拋出要好)。

通過使用ofNullable,你可以創(chuàng)建一個可能包含null引用的Optional對象:

Optional sc = Optional.ofNullable(soundcard);

如果聲卡是null 引用,Optional對象就是一個空的。

對Optional中的值的處理

既然現(xiàn)在已經有了Optional對象,你可以調用相應的方法來處理Optional對象中的值是否存在。和進行null檢測相比,我們可以使用ifPresent()方法,像下面這樣:

Optional soundcard = ...;
soundcard.ifPresent(System.out::println);

這樣就不必再做null檢測,如果Optional對象是空的,那么什么信息將不會打印出來。

你也可以使用isPresent()方法查看Optional對象是否真的存在。另外,還有一個get()方法可以返回Optional對象中的包含的值,如果存在的話。否則會拋出一個NoSuchElementException異常。這兩個方式可以像下面這樣搭配起來使用,從而避免異常:

if(soundcard.isPresent()){
 System.out.println(soundcard.get());
}

但是這種方式不推薦使用(它和null檢測相比沒有什么改進),下面我們將會探討一下工作慣用的方式。

返回默認值和相關操作

當遇到null時一個常規(guī)的操作就是返回一個默認值,你可以使用三元表達式來實現(xiàn):

Soundcard soundcard = maybeSoundcard != null ? maybeSoundcard : new Soundcard("basic_sound_card");

使用Optional對象的話,你可以orElse()使用重寫,當Optional是空的時候orElse()可以返回一個默認值:

Soundcard soundcard = maybeSoundcard.orElse(new Soundcard("defaut"));

類似地,當Optional為空的時候也可以使用orElseThrow()拋出異常:

Soundcard soundcard = 
 maybeSoundCard.orElseThrow(IllegalStateException::new);

使用filter過濾特定的值

我們常常會調用一個對象的方法來判斷它的一下屬性。比如,你可能需要檢測USB端口號是否是某個特定值。為了安全起見,你需要檢查指向USB的醫(yī)用是否是null,然后再調用getVersion()方法,像下面這樣:

USB usb = ...;
if(usb != null && "3.0".equals(usb.getVersion())){
 System.out.println("ok");
}

如果使用Optional的話可以使用filter函數重寫:

Optional maybeUSB = ...;
maybeUSB.filter(usb -> "3.0".equals(usb.getVersion())
     .ifPresent(() -> System.out.println("ok"));

filter方法需要一個predicate對向作為參數。如果Optional中的值存在并且滿足predicate,那么filter函數將會返回滿足條件的值;否則,會返回一個空的Optional對象。

使用map方法進行數據的提取和轉化

一個常見的模式是提取一個對象的一些屬性。比如,對于一個Soundcard對象,你可能需要獲取它的USB對象,然后判斷它的的版本號。通常我們的實現(xiàn)方式是這樣的:

if(soundcard != null){
 USB usb = soundcard.getUSB();
 if(usb != null && "3.0".equals(usb.getVersion()){
 System.out.println("ok");
 }
}

我們可以使用map方法重寫這種檢測null,然后再提取對象類型的對象。

Optional usb = maybeSoundcard.map(Soundcard::getUSB);

這個和使用stream的map函數式一樣的。使用stream需要給map函數傳遞一個函數作為參數,這個傳遞進來的函數將會應用于stream中的每個元素。當stream時空的時候,什么也不會發(fā)生。

Optional中包含的值將會被傳遞進來的函數轉化(這里是一個從聲卡中獲取USB的函數)。如果Optional對象時空的,那么什么也不會發(fā)生。

然后,我們結合map方法和filter方法過濾掉USB的版本號不是3.0的聲卡。

maybeSoundcard.map(Soundcard::getUSB)
  .filter(usb -> "3.0".equals(usb.getVersion())
  .ifPresent(() -> System.out.println("ok"));

這樣我們的代碼開始變得像有點像開始我們給出的樣子,沒有了null檢測。

使用flatMap函數傳遞Optional對象

現(xiàn)在已經介紹了一個可以使用Optional重構代碼的例子,那么我們應該如何使用安全的方式實現(xiàn)下面代碼呢?

String version = computer.getSoundcard().getUSB().getVersion();

注意上面的代碼都是從一個對象中提取另一個對象,使用map函數可以實現(xiàn)。在前面的文章中我們設置了Computer中包含的是一個Optional對象,Soundcard包含的是一個Optional對象,因此我們可以這么重構代碼

String version = computer.map(Computer::getSoundcard)
     .map(Soundcard::getUSB)
     .map(USB::getVersion)
     .orElse("UNKNOWN");

不幸的是,上面的代碼會編譯錯誤,那么為什么呢?computer變量是Optional類型的,所以它調用map函數是沒有問題的。但是getSoundcard()方法返回的是一個Optional的對象,返回的是Optional>類型的對象,進行了第二次map函數的調用,結果調用getUSB()函數就變成非法的了。

下面的圖描述了這種場景:

利用Java8 Optional避免空指針異常的示例

map函數的源碼實現(xiàn)是這樣的:

 public Optional map(Function mapper) {
  Objects.requireNonNull(mapper);
  if (!isPresent())
   return empty();
  else {
   return Optional.ofNullable(mapper.apply(value));
  }
 }

可以看出map函數還會再調用一次Optional.ofNullable() , 從而導致返回Optional>
Optional提供了flatMap這個函數,它的設計意圖是當對Optional對象的值進行轉化(就像map操作)然后一個兩級Optional壓縮成一個。下面的圖展示了Optional對象通過調用map和flatMap進行類型轉化的不同:

利用Java8 Optional避免空指針異常的示例

因此我們可以這樣寫:

String version = computer.flatMap(Computer::getSoundcard)
     .flatMap(Soundcard::getUSB)
     .map(USB::getVersion)
     .orElse("UNKNOWN");

第一個flatMap保證了返回的是Optional而不是Optional> ,第二個flatMap實現(xiàn)了同樣的功能從而返回的是 Optional 。注意第三次調用了map() ,因為getVersion()返回的是一個String對象而不是一個Optional對象。

我們終于把剛開始使用的嵌套null檢查的丑陋代碼改寫了可讀性高的代碼,也避免了空指針異常的出現(xiàn)的代碼。

看完了這篇文章,相信你對“利用Java8 Optional避免空指針異常的示例”有了一定的了解,如果想了解更多相關知識,歡迎關注創(chuàng)新互聯(lián)行業(yè)資訊頻道,感謝各位的閱讀!


網站欄目:利用Java8Optional避免空指針異常的示例
轉載來于:http://weahome.cn/article/goocpp.html

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部