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

Java中空指針異常的幾種解決方案

 更新時(shí)間:2023年01月06日 10:55:29   作者:張吉Jerry  
這篇文章主要介紹了Java中空指針異常的幾種解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教

Java 中任何對(duì)象都有可能為空,當(dāng)我們調(diào)用空對(duì)象的方法時(shí)就會(huì)拋出 NullPointerException 空指針異常,這是一種非常常見的錯(cuò)誤類型。

我們可以使用若干種方法來避免產(chǎn)生這類異常,使得我們的代碼更為健壯。

本文將列舉這些解決方案,包括傳統(tǒng)的空值檢測、編程規(guī)范、以及使用現(xiàn)代 Java 語言引入的各類工具來作為輔助。

運(yùn)行時(shí)檢測

最顯而易見的方法就是使用 if (obj == null) 來對(duì)所有需要用到的對(duì)象來進(jìn)行檢測,包括函數(shù)參數(shù)、返回值、以及類實(shí)例的成員變量。

當(dāng)你檢測到 null 值時(shí),可以選擇拋出更具針對(duì)性的異常類型,如 IllegalArgumentException,并添加消息內(nèi)容。

我們可以使用一些庫函數(shù)來簡化代碼,如 Java 7 開始提供的 Objects#requireNonNull 方法:

public void testObjects(Object arg) {
  Object checked = Objects.requireNonNull(arg, "arg must not be null");
  checked.toString();
}

Guava 的 Preconditions 類中也提供了一系列用于檢測參數(shù)合法性的工具函數(shù),其中就包含空值檢測:

public void testGuava(Object arg) {
  Object checked = Preconditions.checkNotNull(arg, "%s must not be null", "arg");
  checked.toString();
}

我們還可以使用 Lombok 來生成空值檢測代碼,并拋出帶有提示信息的空指針異常:

public void testLombok(@NonNull Object arg) {
  arg.toString();
}

生成的代碼如下:

public void testLombokGenerated(Object arg) {
  if (arg == null) {
    throw new NullPointerException("arg is marked @NonNull but is null");
  }
  arg.toString();
}

這個(gè)注解還可以用在類實(shí)例的成員變量上,所有的賦值操作會(huì)自動(dòng)進(jìn)行空值檢測。

編程規(guī)范

通過遵守某些編程規(guī)范,也可以從一定程度上減少空指針異常的發(fā)生。

使用那些已經(jīng)對(duì) null 值做過判斷的方法,如 String#equalsString#valueOf、以及三方庫中用來判斷字符串和集合是否為空的函數(shù):

if (str != null && str.equals("text")) {}
if ("text".equals(str)) {}

if (obj != null) { obj.toString(); }
String.valueOf(obj); // "null"

// from spring-core
StringUtils.isEmpty(str);
CollectionUtils.isEmpty(col);
// from guava
Strings.isNullOrEmpty(str);
// from commons-collections4
CollectionUtils.isEmpty(col);

如果函數(shù)的某個(gè)參數(shù)可以接收 null 值,考慮改寫成兩個(gè)函數(shù),使用不同的函數(shù)簽名,這樣就可以強(qiáng)制要求每個(gè)參數(shù)都不為空了:

public void methodA(Object arg1) {
  methodB(arg1, new Object[0]);
}

public void methodB(Object arg1, Object[] arg2) {
  for (Object obj : arg2) {} // no null check
}

如果函數(shù)的返回值是集合類型,當(dāng)結(jié)果為空時(shí),不要返回 null 值,而是返回一個(gè)空的集合;如果返回值類型是對(duì)象,則可以選擇拋出異常。

Spring JdbcTemplate 正是使用了這種處理方式:

// 當(dāng)查詢結(jié)果為空時(shí),返回 new ArrayList<>()
jdbcTemplate.queryForList("SELECT * FROM person");

// 若找不到該條記錄,則拋出 EmptyResultDataAccessException
jdbcTemplate.queryForObject("SELECT age FROM person WHERE id = 1", Integer.class);

// 支持泛型集合
public <T> List<T> testReturnCollection() {
  return Collections.emptyList();
}

靜態(tài)代碼分析

Java 語言有許多靜態(tài)代碼分析工具,如 Eclipse IDE、SpotBugs、Checker Framework 等,它們可以幫助程序員檢測出編譯期的錯(cuò)誤。

結(jié)合 @Nullable@Nonnull 等注解,我們就可以在程序運(yùn)行之前發(fā)現(xiàn)可能拋出空指針異常的代碼。

但是,空值檢測注解還沒有得到標(biāo)準(zhǔn)化。

雖然 2006 年 9 月社區(qū)提出了 JSR 305 規(guī)范,但它長期處于擱置狀態(tài)。

很多第三方庫提供了類似的注解,且得到了不同工具的支持,其中使用較多的有:

  • javax.annotation.Nonnull:由 JSR 305 提出,其參考實(shí)現(xiàn)為 com.google.code.findbugs.jsr305;
  • org.eclipse.jdt.annotation.NonNull:Eclipse IDE 原生支持的空值檢測注解;
  • edu.umd.cs.findbugs.annotations.NonNull:SpotBugs 使用的注解,基于 findbugs.jsr305;
  • org.springframework.lang.NonNull:Spring Framework 5.0 開始提供;
  • org.checkerframework.checker.nullness.qual.NonNull:Checker Framework 使用;
  • android.support.annotation.NonNull:集成在安卓開發(fā)工具中;

我建議使用一種跨 IDE 的解決方案,如 SpotBugs 或 Checker Framework,它們都能和 Maven 結(jié)合得很好。

SpotBugs 與 @NonNull、@CheckForNull

SpotBugs 是 FindBugs 的后繼者。通過在方法的參數(shù)和返回值上添加 @NonNull@CheckForNull 注解,SpotBugs 可以幫助我們進(jìn)行編譯期的空值檢測。

需要注意的是,SpotBugs 不支持 @Nullable 注解,必須用 @CheckForNull 代替。

如官方文檔中所說,僅當(dāng)需要覆蓋 @ParametersAreNonnullByDefault 時(shí)才會(huì)用到 @Nullable。

官方文檔 中說明了如何將 SpotBugs 應(yīng)用到 Maven 和 Eclipse 中去。我們還需要將 spotbugs-annotations 加入到項(xiàng)目依賴中,以便使用對(duì)應(yīng)的注解。

<dependency>
    <groupId>com.github.spotbugs</groupId>
    <artifactId>spotbugs-annotations</artifactId>
    <version>3.1.7</version>
</dependency>

以下是對(duì)不同使用場景的說明:

@NonNull
private Object returnNonNull() {
  // 錯(cuò)誤:returnNonNull() 可能返回空值,但其已聲明為 @Nonnull
  return null;
}

@CheckForNull
private Object returnNullable() {
  return null;
}

public void testReturnNullable() {
  Object obj = returnNullable();
  // 錯(cuò)誤:方法的返回值可能為空
  System.out.println(obj.toString());
}

private void argumentNonNull(@NonNull Object arg) {
  System.out.println(arg.toString());
}

public void testArgumentNonNull() {
  // 錯(cuò)誤:不能將 null 傳遞給非空參數(shù)
  argumentNonNull(null);
}

public void testNullableArgument(@CheckForNull Object arg) {
  // 錯(cuò)誤:參數(shù)可能為空
  System.out.println(arg.toString());
}

對(duì)于 Eclipse 用戶,還可以使用 IDE 內(nèi)置的空值檢測工具,只需將默認(rèn)的注解 org.eclipse.jdt.annotation.Nullable 替換為 SpotBugs 的注解即可:

Eclipse null analysis

Checker Framework 與 @NonNull、@Nullable

Checker Framework 能夠作為 javac 編譯器的插件運(yùn)行,對(duì)代碼中的數(shù)據(jù)類型進(jìn)行檢測,預(yù)防各類問題。

我們可以參照 官方文檔,將 Checker Framework 與 maven-compiler-plugin 結(jié)合,之后每次執(zhí)行 mvn compile 時(shí)就會(huì)進(jìn)行檢查。

Checker Framework 的空值檢測程序支持幾乎所有的注解,包括 JSR 305、Eclipse、甚至 lombok.NonNull

import org.checkerframework.checker.nullness.qual.Nullable;

@Nullable
private Object returnNullable() {
  return null;
}

public void testReturnNullable() {
  Object obj = returnNullable();
  // 錯(cuò)誤:obj 可能為空
  System.out.println(obj.toString());
}

Checker Framework 默認(rèn)會(huì)將 @NonNull 應(yīng)用到所有的函數(shù)參數(shù)和返回值上,因此,即使不添加這個(gè)注解,以下程序也是無法編譯通過的:

private Object returnNonNull() {
  // 錯(cuò)誤:方法聲明為 @NonNull,但返回的是 null。
  return null;
}

private void argumentNonNull(Object arg) {
  System.out.println(arg.toString());
}

public void testArgumentNonNull() {
  // 錯(cuò)誤:參數(shù)聲明為 @NonNull,但傳入的是 null。
  argumentNonNull(null);
}

Checker Framework 對(duì)使用 Spring Framework 5.0 以上的用戶非常有用,因?yàn)?Spring 提供了內(nèi)置的空值檢測注解,且能夠被 Checker Framework 支持。

一方面我們無需再引入額外的 Jar 包,更重要的是 Spring Framework 代碼本身就使用了這些注解,這樣我們?cè)谡{(diào)用它的 API 時(shí)就能有效地處理空值了。

舉例來說,StringUtils 類里可以傳入空值的函數(shù)、以及會(huì)返回空值的函數(shù)都添加了 @Nullable 注解,而未添加的方法則繼承了整個(gè)框架的 @NonNull 注解,因此,下列代碼中的空指針異常就可以被 Checker Framework 檢測到了:

// 這是 spring-core 中定義的類和方法
public abstract class StringUtils {
  // str 參數(shù)繼承了全局的 @NonNull 注解
  public static String capitalize(String str) {}

  @Nullable
  public static String getFilename(@Nullable String path) {}
}

// 錯(cuò)誤:參數(shù)聲明為 @NonNull,但傳入的是 null。
StringUtils.capitalize(null);

String filename = StringUtils.getFilename("/path/to/file");
// 錯(cuò)誤:filename 可能為空。
System.out.println(filename.length());

Optional 類型

Java 8 引入了 Optional<T> 類型,我們可以用它來對(duì)函數(shù)的返回值進(jìn)行包裝。

這種方式的優(yōu)點(diǎn)是可以明確定義該方法是有可能返回空值的,因此調(diào)用方必須做好相應(yīng)處理,這樣也就不會(huì)引發(fā)空指針異常。

但是,也不可避免地需要編寫更多代碼,而且會(huì)產(chǎn)生很多垃圾對(duì)象,增加 GC 的壓力,因此在使用時(shí)需要酌情考慮。

Optional<String> opt;

// 創(chuàng)建
opt = Optional.empty();
opt = Optional.of("text");
opt = Optional.ofNullable(null);

// 判斷并讀取
if (opt.isPresent()) {
  opt.get();
}

// 默認(rèn)值
opt.orElse("default");
opt.orElseGet(() -> "default");
opt.orElseThrow(() -> new NullPointerException());

// 相關(guān)操作
opt.ifPresent(value -> {
  System.out.println(value);
});
opt.filter(value -> value.length() > 5);
opt.map(value -> value.trim());
opt.flatMap(value -> {
  String trimmed = value.trim();
  return trimmed.isEmpty() ? Optional.empty() : Optional.of(trimmed);
});

方法的鏈?zhǔn)秸{(diào)用很容易引發(fā)空指針異常,但如果返回值都用 Optional 包裝起來,就可以用 flatMap 方法來實(shí)現(xiàn)安全的鏈?zhǔn)秸{(diào)用了:

String zipCode = getUser()
    .flatMap(User::getAddress)
    .flatMap(Address::getZipCode)
    .orElse("");

Java 8 Stream API 同樣使用了 Optional 作為返回類型:

stringList.stream().findFirst().orElse("default");
stringList.stream()
    .max(Comparator.naturalOrder())
    .ifPresent(System.out::println);

此外,Java 8 還針對(duì)基礎(chǔ)類型提供了單獨(dú)的 Optional 類,如 OptionalInt、OptionalDouble 等,在性能要求比較高的場景下很適用。

其它 JVM 語言中的空指針異常

Scala 語言中的 Option 類可以對(duì)標(biāo) Java 8 的 Optional

它有兩個(gè)子類型,Some 表示有值,None 表示空。

val opt: Option[String] = Some("text")
opt.getOrElse("default")

除了使用 Option#isEmpty 判斷,還可以使用 Scala 的模式匹配:

opt match {
  case Some(text) => println(text)
  case None => println("default")
}

Scala 的集合處理函數(shù)庫非常強(qiáng)大,Option 則可直接作為集合進(jìn)行操作,如 filermap、以及列表解析(for-comprehension):

opt.map(_.trim).filter(_.length > 0).map(_.toUpperCase).getOrElse("DEFAULT")
val upper = for {
  text <- opt
  trimmed <- Some(text.trim())
  upper <- Some(trimmed) if trimmed.length > 0
} yield upper
upper.getOrElse("DEFAULT")

Kotlin 使用了另一種方式,用戶在定義變量時(shí)就需要明確區(qū)分 可空和不可空類型。當(dāng)可空類型被使用時(shí),就必須進(jìn)行空值檢測。

var a: String = "text"
a = null // 錯(cuò)誤:無法將 null 賦值給非空 String 類型。

val b: String? = "text"
// 錯(cuò)誤:操作可空類型時(shí)必須使用安全操作符(?.)或強(qiáng)制忽略(!!.)。
println(b.length)

val l: Int? = b?.length // 安全操作
b!!.length // 強(qiáng)制忽略,可能引發(fā)空值異常

Kotlin 的特性之一是與 Java 的可互操作性,但 Kotlin 編譯器無法知曉 Java 類型是否為空,這就需要在 Java 代碼中使用注解了,而 Kotlin 支持的 注解 也非常廣泛。

Spring Framework 5.0 起原生支持 Kotlin,其空值檢測也是通過注解進(jìn)行的,使得 Kotlin 可以安全地調(diào)用 Spring Framework 的所有 API。

結(jié)論

在以上這些方案中,我比較推薦使用注解來預(yù)防空指針異常,因?yàn)檫@種方式十分有效,對(duì)代碼的侵入性也較小。

所有的公共 API 都應(yīng)該使用 @Nullable@NonNull 進(jìn)行注解,這樣就能強(qiáng)制調(diào)用方對(duì)空指針異常進(jìn)行預(yù)防,讓我們的程序更為健壯。希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。

參考資料

https://howtodoinjava.com/java/exception-handling/how-to-effectively-handle-nullpointerexception-in-java/

http://jmri.sourceforge.net/help/en/html/doc/Technical/SpotBugs.shtml

https://dzone.com/articles/features-to-avoid-null-reference-exceptions-java-a

相關(guān)文章

  • Spring Boot實(shí)戰(zhàn)之靜態(tài)資源處理

    Spring Boot實(shí)戰(zhàn)之靜態(tài)資源處理

    這篇文章主要介紹了Spring Boot實(shí)戰(zhàn)之靜態(tài)資源處理,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2018-01-01
  • Java復(fù)合語句的使用方法詳解

    Java復(fù)合語句的使用方法詳解

    這篇文章主要介紹了Java編程中復(fù)合語句,結(jié)合相關(guān)的具體實(shí)例介紹了其用法,需要的朋友可以參考下
    2017-09-09
  • lambdaQueryWrapper多條件嵌套查詢方式

    lambdaQueryWrapper多條件嵌套查詢方式

    這篇文章主要介紹了lambdaQueryWrapper多條件嵌套查詢方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教。
    2022-01-01
  • Java static方法用法實(shí)戰(zhàn)案例總結(jié)

    Java static方法用法實(shí)戰(zhàn)案例總結(jié)

    這篇文章主要介紹了Java static方法用法,結(jié)合具體案例形式總結(jié)分析了java static方法功能、使用方法及相關(guān)操作注意事項(xiàng),需要的朋友可以參考下
    2019-09-09
  • Jackson使用示例-Bean、XML、Json之間相互轉(zhuǎn)換

    Jackson使用示例-Bean、XML、Json之間相互轉(zhuǎn)換

    Jackson是一個(gè)強(qiáng)大工具,可用于Json、XML、實(shí)體之間的相互轉(zhuǎn)換,JacksonXmlElementWrapper用于指定List等集合類,外圍標(biāo)簽名,JacksonXmlProperty指定包裝標(biāo)簽名,或者指定標(biāo)簽內(nèi)部屬性名,JacksonXmlRootElement指定生成xml根標(biāo)簽的名字,JacksonXmlText指定當(dāng)前這個(gè)值
    2024-05-05
  • JAVA API 實(shí)用類 String詳解

    JAVA API 實(shí)用類 String詳解

    這篇文章主要介紹了java String的深入理解的相關(guān)資料,希望通過本文大家能理解String的用法,需要的朋友可以參考下
    2021-10-10
  • 學(xué)習(xí)Java之IO流的基礎(chǔ)概念詳解

    學(xué)習(xí)Java之IO流的基礎(chǔ)概念詳解

    這篇文章主要給大家介紹了Java中的IO流,我們首先要搞清楚一件事,就是為什么需要IO流這個(gè)東西,但在正式學(xué)習(xí)IO流的使用之前,小編有必要帶大家先了解一下IO流的基本概念,需要的朋友可以參考下
    2023-09-09
  • 大數(shù)據(jù) java hive udf函數(shù)的示例代碼(手機(jī)號(hào)碼脫敏)

    大數(shù)據(jù) java hive udf函數(shù)的示例代碼(手機(jī)號(hào)碼脫敏)

    這篇文章主要介紹了大數(shù)據(jù) java hive udf函數(shù)(手機(jī)號(hào)碼脫敏),的相關(guān)知識(shí),本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-06-06
  • Spring中@Import的各種用法以及ImportAware接口詳解

    Spring中@Import的各種用法以及ImportAware接口詳解

    這篇文章主要介紹了Spring中@Import的各種用法以及ImportAware接口詳解,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-10-10
  • mybatis-generator如何自定義注釋生成

    mybatis-generator如何自定義注釋生成

    這篇文章主要介紹了mybatis-generator如何自定義注釋生成的操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-09-09

最新評(píng)論