欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

利用Java8 Optional類優(yōu)雅如何地解決空指針問題

 更新時(shí)間:2020年11月17日 16:43:51   作者:愛笑的架構(gòu)師  
這篇文章主要給大家介紹了關(guān)于如何利用Java8 Optional類優(yōu)雅解決空指針問題的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

前言

Java8 由Oracle在2014年發(fā)布,是繼Java5之后最具革命性的版本。

Java8吸收其他語言的精髓帶來了函數(shù)式編程,lambda表達(dá)式,Stream流等一系列新特性,學(xué)會(huì)了這些新特性,可以讓你實(shí)現(xiàn)高效編碼優(yōu)雅編碼。

1. 不受待見的空指針異常

有個(gè)小故事:null引用最早是由英國科學(xué)家Tony Hoare提出的,多年后Hoare為自己的這個(gè)想法感到后悔莫及,并認(rèn)為這是"價(jià)值百萬的重大失誤"??梢娍罩羔樖嵌嗝床皇艽姟?/p>

NullPointerException是Java開發(fā)中最常遇見的異常,遇到這種異常我們通常的解決方法是在調(diào)用的地方加一個(gè)if判空。

if判空越多會(huì)造成過多的代碼分支,后續(xù)代碼維護(hù)也就越來越復(fù)雜。

2. 糟糕的代碼

比如看下面這個(gè)例子,使用過多的if判空。

Person對(duì)象里定義了House對(duì)象,House對(duì)象里定義了Address對(duì)象:

public class Person {
 private String name;
 private int age;
 private House house;

 public House getHouse() {
  return house;
 }
}

class House {
 private long price;
 private Address address;

 public Address getAddress() {
  return address;
 }
}

class Address {
 private String country;
 private String city;

 public String getCity() {
  return city;
 }
}

現(xiàn)在獲取這個(gè)人買房的城市,那么通常會(huì)這樣寫:

public String getCity() {
 String city = new Person().getHouse().getAddress().getCity();
 return city;
}

但是這樣寫容易出現(xiàn)空指針的問題,比如這個(gè)人沒有房,House對(duì)象為null。接著你會(huì)改造這段代碼,加上很多判斷條件:

public String getCity2(Person person) {
 if (person != null) {
  House house = person.getHouse();
  if (house != null) {
   Address address = house.getAddress();
   if (address != null) {
    String city = address.getCity();
    return city;
   }
  }
 }
 return "unknown";
}

為了避免空指針異常,每一層都加上判斷,但是這樣會(huì)造成代碼嵌套太深,不易維護(hù)。

你可能想到如何改造上面的代碼,比如加上提前判空退出:

public String getCity3(Person person) {
 String city = "unknown";
 if (person == null) {
  return city; 
 }

 House house = person.getHouse();
 if (house == null) {
  return city;
 }

 Address address = house.getAddress();
 if (address == null) {
  return city;
 }

 return address.getCity();
}

但是這樣簡單的代碼已經(jīng)加入了三個(gè)退出條件,非常不利于后面代碼維護(hù)。那怎樣才能將代碼寫的優(yōu)雅一點(diǎn)呢,下面引入今天的主角"Optional"。

3. 解決空指針的"銀彈"

從Java8開始引入了一個(gè)新類 java.util.Optional,這是一個(gè)對(duì)象的容器,意味著可能包含或者沒有包含一個(gè)非空的值。下面重點(diǎn)看一下Optional的常用方法:

public final class Optional<T> {
 // 通過指定非空值創(chuàng)建Optional對(duì)象
 // 如果指定的值為null,會(huì)拋空指針異常
 public static <T> Optional<T> of(T value) {
  return new Optional<>(value);
 }
 
 // 通過指定可能為空的值創(chuàng)建Optional對(duì)象
 public static <T> Optional<T> ofNullable(T value) {
  return value == null ? empty() : of(value);
 }

 // 返回值,不存在拋異常
 public T get() {
  if (value == null) {
   throw new NoSuchElementException("No value present");
  }
  return value;
 }
 
 // 如果值存在,根據(jù)consumer實(shí)現(xiàn)類消費(fèi)該值
 public void ifPresent(Consumer<? super T> consumer) {
  if (value != null)
   consumer.accept(value);
 }
 
 // 如果值存在則返回,如果值為空則返回指定的默認(rèn)值
 public T orElse(T other) {
  return value != null ? value : other;
 }

 // map flatmap等方法與Stream使用方法類似,這里不再贅述,讀者可以參考之前的Stream系列。
}

以上就是Optional類常用的方法,使用起來非常簡單。

4. Optional使用入門

(1)創(chuàng)建Optional實(shí)例

創(chuàng)建空的Optional對(duì)象。可以通過靜態(tài)工廠方法Optional.Empty() 創(chuàng)建一個(gè)空的對(duì)象,例如:

Optional<Person> optionalPerson = Optional.Empty();

指定非空值創(chuàng)建Optional對(duì)象。

Person person = new Person();
Optional<Person> optionalPerson = Optional.of(person);

指定可能為空的值創(chuàng)建Optional對(duì)象。

Person person = null; // 可能為空
Optional<Person> optionalPerson = Optional.of(person);

(2)常用方法

ifPresent

如果值存在,則調(diào)用consumer實(shí)例消費(fèi)該值,否則什么都不執(zhí)行。舉個(gè)栗子:

String str = "hello java8";
// output: hello java8
Optional.ofNullable(str).ifPresent(System.out::println);

String str2 = null;
// output: nothing
Optional.ofNullable(str2).ifPresent(System.out::println);

filter, map, flatMap

在三個(gè)方法在前面講Stream的時(shí)候已經(jīng)詳細(xì)講解過,讀者可以翻看之前寫的文章,這里不再贅述。

orElse

如果value為空,則返回默認(rèn)值,舉個(gè)栗子:

public void test(String city) {
 String defaultCity = Optional.ofNullable(city).orElse("unknown");
}

orElseGet

如果value為空,則調(diào)用Supplier實(shí)例返回一個(gè)默認(rèn)值。舉個(gè)例子:

public void test2(String city) {
 // 如果city為空,則調(diào)用generateDefaultCity方法
 String defaultCity = Optional.of(city).orElseGet(this::generateDefaultCity);
}

private String generateDefaultCity() {
 return "beijing";
}

orElseThrow

如果value為空,則拋出自定義異常。舉個(gè)栗子:

public void test3(String city) {
 // 如果city為空,則拋出空指針異常。
 String defaultCity = Optional.of(city).orElseThrow(NullPointerException::new);
}

5. 使用Optional重構(gòu)代碼

再看一遍重構(gòu)之前的代碼,使用了三個(gè)if使代碼嵌套層次變得很深。

// before refactor
public String getCity2(Person person) {
 if (person != null) {
  House house = person.getHouse();
  if (house != null) {
   Address address = house.getAddress();
   if (address != null) {
    String city = address.getCity();
    return city;
   }
  }
 }
 return "unknown";
}

使用Optional重構(gòu)

public String getCityUsingOptional(Person person) {
 String city = Optional.ofNullable(person)
   .map(Person::getHouse)
   .map(House::getAddress)
   .map(Address::getCity).orElse("Unknown city");
 return city;
}

只使用了一行代碼就獲取到city值,不用再去不斷的判斷是否為空,這樣寫代碼是不是很優(yōu)雅呀。趕緊用Optional重構(gòu)你的項(xiàng)目吧~

總結(jié)

到此這篇關(guān)于利用Java8 Optional類優(yōu)雅如何地解決空指針問題的文章就介紹到這了,更多相關(guān)Java8 Optional類解決空指針內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • java中如何使用HttpClient調(diào)用接口

    java中如何使用HttpClient調(diào)用接口

    這篇文章主要介紹了java中如何使用HttpClient調(diào)用接口,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-10-10
  • Java中數(shù)組的一些常見操作和技巧分析

    Java中數(shù)組的一些常見操作和技巧分析

    這篇文章主要給大家介紹了關(guān)于Java中數(shù)組的一些常見操作和技巧分析的相關(guān)資料,數(shù)組(Array)是Java中的一種引用數(shù)據(jù)類型,是多個(gè)相同類型數(shù)據(jù)一定順序排列的集合,并使用一個(gè)名字命名,并通過編號(hào)的方式對(duì)這些數(shù)據(jù)進(jìn)行統(tǒng)一管理,需要的朋友可以參考下
    2023-08-08
  • 使用IDEA異常斷點(diǎn)來定位java.lang.ArrayStoreException的問題

    使用IDEA異常斷點(diǎn)來定位java.lang.ArrayStoreException的問題

    這篇文章主要介紹了使用IDEA異常斷點(diǎn)來定位java.lang.ArrayStoreException的問題,平常開發(fā)過程中面對(duì)這種描述不夠清楚,無法定位具體原因的問題該如何處理,下面我們來一起學(xué)習(xí)一下吧
    2019-06-06
  • Java實(shí)現(xiàn)文件分割和文件合并實(shí)例

    Java實(shí)現(xiàn)文件分割和文件合并實(shí)例

    本篇文章主要介紹了Java實(shí)現(xiàn)文件分割和文件合并實(shí)例,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2017-08-08
  • Java?synchronized與死鎖深入探究

    Java?synchronized與死鎖深入探究

    這篇文章主要介紹了Java?synchronized與死鎖,Java中提供了synchronized關(guān)鍵字,將可能引發(fā)安全問題的代碼包裹在synchronized代碼塊中,表示這些代碼需要進(jìn)行線程同步
    2023-01-01
  • 詳解使用JavaCV/OpenCV抓取并存儲(chǔ)攝像頭圖像

    詳解使用JavaCV/OpenCV抓取并存儲(chǔ)攝像頭圖像

    本篇文章主要介紹了使用JavaCV/OpenCV抓取并存儲(chǔ)攝像頭圖像,實(shí)例分析了使用JavaCV/OpenCV抓取并存儲(chǔ)攝像頭圖像的技巧,非常具有實(shí)用價(jià)值,需要的朋友可以參考下
    2017-04-04
  • JAVA十大排序算法之堆排序詳解

    JAVA十大排序算法之堆排序詳解

    這篇文章主要介紹了java中的冒泡排序,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考
    2021-08-08
  • jsp、struts、spring、mybatis實(shí)現(xiàn)前端頁面功能模塊化拆分的方案

    jsp、struts、spring、mybatis實(shí)現(xiàn)前端頁面功能模塊化拆分的方案

    這篇文章主要介紹了 jsp、struts、spring、mybatis實(shí)現(xiàn)前端頁面功能模塊化拆分的方案,非常不錯(cuò),需要的朋友參考下
    2017-01-01
  • Java8新特性之Lambda表達(dá)式的使用

    Java8新特性之Lambda表達(dá)式的使用

    這篇文章主要介紹了Java8新特性之Lambda表達(dá)式的使用,文中有非常詳細(xì)的代碼示例,對(duì)正在學(xué)習(xí)java的小伙伴們有非常好的幫助,需要的朋友可以參考下
    2021-04-04
  • java?executor包參數(shù)處理功能?

    java?executor包參數(shù)處理功能?

    這篇文章主要介紹了java?executor包參數(shù)處理功能,sql語句中的參數(shù)賦值是有由executor包中的parameter子包完成的。parameter子包其實(shí)只有一個(gè)parameterHandler接口并且它定義了兩個(gè)方法,下面我們就來看詳細(xì)內(nèi)容吧,需要的朋友可以參考一下
    2022-02-02

最新評(píng)論