java.lang.NullPointerException 如何處理空指針異常的實(shí)現(xiàn)
當(dāng)應(yīng)用程序試圖null在需要對(duì)象的情況下使用時(shí)拋出。這些包括:
- 調(diào)用null對(duì)象的實(shí)例方法。
- 訪問或修改null對(duì)象的字段。
- 把長(zhǎng)度null當(dāng)作一個(gè)數(shù)組。
- 像訪問或修改null陣列一樣訪問或修改插槽。
- 投擲null就好像它是一個(gè)Throwable 價(jià)值。
- 應(yīng)用程序應(yīng)該拋出此類的實(shí)例來(lái)指示null對(duì)象的其他非法使用。
- NullPointerException對(duì)象可以由虛擬機(jī)構(gòu)造,就像抑制被禁用和/或堆棧跟蹤不可寫一樣。
為什么我們需要空值?
- 如前所述,nullJava是一種特殊的值。
- 它在編碼某些設(shè)計(jì)模式(如空對(duì)象模式和單例模式)時(shí)非常有用。
- 空對(duì)象模式提供了一個(gè)對(duì)象作為缺少給定類型對(duì)象的代理。
- Singleton模式確保只創(chuàng)建一個(gè)類的一個(gè)實(shí)例,并且旨在提供對(duì)象的全局訪問點(diǎn)。
例如,最多創(chuàng)建一個(gè)類實(shí)例的示例方法是將其所有構(gòu)造函數(shù)聲明為private,然后創(chuàng)建一個(gè)返回該類的唯一實(shí)例的公共方法:
TestSingleton.java:
import java.util.UUID;
class Singleton {
private static Singleton single = null;
private String ID = null;
private Singleton() {
/* Make it private, in order to prevent the creation of new instances of
* the Singleton class. */
ID = UUID.randomUUID().toString(); // Create a random ID.
}
public static Singleton getInstance() {
if (single == null)
single = new Singleton();
return single;
}
public String getID() {
return this.ID;
}
}
public class TestSingleton {
public static void main(String[] args) {
Singleton s = Singleton.getInstance();
System.out.println(s.getID());
}
}
在這個(gè)例子中,我們聲明了一個(gè)Singleton類的靜態(tài)實(shí)例。該實(shí)例在該getInstance方法內(nèi)最多初始化一次。注意使用null啟用唯一實(shí)例創(chuàng)建的值。
如何避免NullPointerException
為了避免這種情況NullPointerException,請(qǐng)確保在使用它們之前,所有對(duì)象都已正確初始化。注意,當(dāng)你聲明一個(gè)引用變量時(shí),你真的創(chuàng)建了一個(gè)指向?qū)ο蟮闹羔槨T谙驅(qū)ο笳?qǐng)求方法或字段之前,您必須驗(yàn)證指針是否為空。
另外,如果引發(fā)異常,請(qǐng)使用駐留在異常堆棧跟蹤中的信息。執(zhí)行的堆棧跟蹤由JVM提供,以啟用應(yīng)用程序的調(diào)試。找到捕獲異常的方法和行,然后確定哪個(gè)引用等于在特定行中為null。
在本節(jié)的其余部分中,我們將介紹一些處理上述例外的技術(shù)。但是,它們并沒有消除這個(gè)問題,程序員在編寫應(yīng)用程序時(shí)應(yīng)該小心。
1.字符串與文字的比較
應(yīng)用程序執(zhí)行代碼中的一個(gè)非常常見的情況涉及字符串變量和文字之間的比較。文字可以是一個(gè)字符串或Enum的元素。不要從空對(duì)象調(diào)用方法,而應(yīng)考慮從文字中調(diào)用它。例如,觀察以下情況:
String str = null;
if(str.equals(“Test”)){
/ *這里的代碼將不會(huì)被觸發(fā),因?yàn)闀?huì)拋出異常。* /
}
上面的代碼片段會(huì)拋出一個(gè)NullPointerException。但是,如果我們從文字中調(diào)用方法,那么執(zhí)行流程通常會(huì)繼續(xù):
String str = null;
if(“Test”.equals(str)){
/ *正確的用例。不會(huì)拋出異常。* /
}
2.檢查方法的參數(shù)
在執(zhí)行你自己的方法的主體之前,一定要檢查它的參數(shù)為空值。只有在正確檢查了參數(shù)后,才繼續(xù)執(zhí)行該方法。否則,您可以拋出一個(gè)IllegalArgumentException并通知調(diào)用方法傳遞的參數(shù)有問題。
例如:
public static int getLength(String s){
如果(s == null)
拋出新的IllegalArgumentException(“參數(shù)不能為空”);
return s.length();
}
3.優(yōu)先使用String.valueOf()方法代替toString()
當(dāng)您的應(yīng)用程序代碼需要對(duì)象的字符串表示形式時(shí),請(qǐng)避免使用該對(duì)象的toString方法。如果你的對(duì)象的引用等于null,NullPointerException則會(huì)拋出a。
相反,考慮使用靜態(tài)String.valueOf方法,該方法不會(huì)拋出任何異常并打印"null",以防函數(shù)的參數(shù)等于null。
4.使用三元運(yùn)算符
該ternary操作是非常有用的,可以幫助我們避免了NullPointerException。運(yùn)營(yíng)商的形式是:
布爾表達(dá)式?value1:value2;
首先,評(píng)估布爾表達(dá)式。如果表達(dá)式為true,則返回value1,否則返回value2。我們可以使用ternary運(yùn)算符來(lái)處理空指針,如下所示:
String message =(str == null)?"":str.substring(0,10);
如果str引用為空,則消息變量將為空。否則,如果str指向?qū)嶋H數(shù)據(jù),則消息將檢索它的前10個(gè)字符。
5.創(chuàng)建返回空集合而不是null的方法
一個(gè)非常好的技術(shù)是創(chuàng)建返回一個(gè)空集合的方法,而不是一個(gè)null值。你的應(yīng)用程序的代碼可以遍歷空集合并使用它的方法和字段,而不會(huì)拋出一個(gè)NullPointerException。例如:
Example.java
public class Example {
private static List<Integer> numbers = null;
public static List<Integer> getList() {
if (numbers == null)
return Collections.emptyList();
else
return numbers;
}
}
6.使用Apache的StringUtils類
Apache的Commons Lang是一個(gè)為java.langAPI 提供幫助工具的庫(kù),比如字符串操作方法。提供字符串操作的示例類是StringUtils.java,它null靜靜地處理輸入字符串。
你可以使用StringUtils.isNotEmpty, StringUtils.IsEmpty和StringUtils.equals方法,以避免NullPointerException。例如:
if(StringUtils.isNotEmpty(str)){
System.out.println(str.toString());
}
7.使用contains(),containsKey(),containsValue()方法
如果您的應(yīng)用程序代碼使用集合,例如Maps考慮使用包含containsKey和containsValue方法。例如,在地圖中驗(yàn)證其存在之后,檢索特定鍵的值:
Map <String,String> map = ... ... String key = ... String value = map.get(key); 的System.out.println(value.toString()); //如果值為null,則會(huì)拋出異常。
在上面的代碼片段中,我們不檢查密鑰是否真的存在于內(nèi)部Map,因此返回的值可以是null。最安全的方法如下:
Map <String,String> map = ...
...
String key = ...
if(map.containsKey(key)){
String value = map.get(key);
的System.out.println(value.toString()); //不會(huì)拋出異常。
}
8.檢查外部方法的返回值
在實(shí)踐中使用外部庫(kù)是很常見的。這些庫(kù)包含返回引用的方法。確保返回的參考不是null。另外,請(qǐng)考慮閱讀該方法的Javadoc,以便更好地理解其功能和返回值。
9.使用斷言
斷言在測(cè)試代碼時(shí)非常有用,并且可以被使用,以避免執(zhí)行代碼片斷,從而導(dǎo)致錯(cuò)誤NullPointerException。Java斷言是用assert關(guān)鍵字實(shí)現(xiàn)的,并拋出一個(gè)AssertionError。
請(qǐng)注意,您必須顯式啟用JVM的斷言標(biāo)志,方法是使用–ea參數(shù)執(zhí)行該標(biāo)志。否則,斷言將被完全忽略。
使用Java斷言的示例示例如下:
public static int getLength(String s){
/ *確保String不為null。* /
assert(s!= null);
return s.length();
}
如果您執(zhí)行上面的代碼段并傳遞一個(gè)空參數(shù)getLength,則會(huì)出現(xiàn)以下錯(cuò)誤消息:
Exception in thread "main" java.lang.AssertionError
最后,您可以使用測(cè)試框架Assert提供的類jUnit。
10.單元測(cè)試
在測(cè)試代碼的功能和正確性時(shí),單元測(cè)試可能非常有用。花一些時(shí)間編寫一些測(cè)試用例,驗(yàn)證NullPointerException應(yīng)用程序的代碼是否經(jīng)歷了特定的執(zhí)行流程,否則將引發(fā)no 。
現(xiàn)有的NullPointerException安全方法
1.訪問類的靜態(tài)成員或方法
當(dāng)你的代碼試圖訪問靜態(tài)變量或類的方法時(shí),即使對(duì)象的引用等于null,JVM也不會(huì)拋出一個(gè)NullPointerException。這是由于Java編譯器在編譯過程中將靜態(tài)方法和字段存儲(chǔ)在特殊位置。因此,靜態(tài)字段和方法不與對(duì)象相關(guān)聯(lián),而與類的名稱相關(guān)聯(lián)。
例如,下面的代碼不會(huì)拋出NullPointerException:
TestStatic.java:
class SampleClass {
public static void printMessage(){
System.out.println(“Hello Java Geeks!”);
}
}
public class TestStatic {
public static void main(String [] args){
SampleClass sc = null;
sc.printMessage();
}
}
注意,盡管SampleClass等于的實(shí)例null將會(huì)被正確執(zhí)行。但是,對(duì)于靜態(tài)方法或字段,最好以靜態(tài)方式訪問它們,比如SampleClass.printMessage()。
2.運(yùn)營(yíng)商的instanceof
instanceof即使對(duì)象的引用等于,也可以使用該運(yùn)算符null。在instanceof操作時(shí),參考值等于為null,不拋出一個(gè)返回false NullPointerException。例如,考慮下面的代碼片段:
String str = null;
if(str instanceof String)
System.out.println("It's an instance of the String class!");
else
System.out.println("Not an instance of the String class!");
正如預(yù)期的那樣,執(zhí)行的結(jié)果是:
Not an instance of the String class!
這是一篇關(guān)于如何處理Java的教程N(yùn)ullPointerException。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- Java空指針異常NullPointerException的原因與解決方案
- 解決java.lang.NullPointerException問題(空指針異常)
- Java如何優(yōu)雅地避免空指針異常(NullPointerException)
- Java中redisTemplate注入失敗NullPointerException異常問題解決
- 詳解Java中NullPointerException異常的原因和解決辦法
- java.lang.NullPointerException異常問題解決方案
- 詳解Java中NullPointerException異常的原因詳解以及解決方法
- Java中NullPointerException的異常解決
相關(guān)文章
200行Java代碼如何實(shí)現(xiàn)依賴注入框架詳解
依賴注入對(duì)大家來(lái)說應(yīng)該都不陌生,下面這篇文章主要給大家介紹了關(guān)于利用200行Java代碼如何實(shí)現(xiàn)依賴注入框架的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),需要的朋友可以參考借鑒,下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2018-05-05
list,set,map,數(shù)組之間的相互轉(zhuǎn)換詳細(xì)解析
以下是對(duì)Java中l(wèi)ist,set,map,數(shù)組之間的相互轉(zhuǎn)換進(jìn)行了詳細(xì)的分析介紹,需要的朋友可以過來(lái)參考下2013-09-09
IDEA配置和啟動(dòng)maven項(xiàng)目詳細(xì)步驟
本文介紹了從SVN檢出Web項(xiàng)目并進(jìn)行Maven化、JDK和項(xiàng)目結(jié)構(gòu)配置、Spring和Tomcat環(huán)境搭建的詳細(xì)步驟,幫助讀者順利完成Java Web項(xiàng)目的開發(fā)環(huán)境搭建2025-10-10
Java實(shí)現(xiàn)Excel圖片URL篩選與大小檢測(cè)的全過程
在數(shù)據(jù)處理場(chǎng)景中,我們常需篩選Excel中的圖片URL,本文分享一個(gè)完整的Java方案,涵蓋從讀取圖片URL到檢測(cè)有效性、篩選大小,再到生成新Excel文件的全過程,同時(shí)講解開發(fā)與優(yōu)化過程,需要的朋友可以參考下2025-08-08
Java SpringMVC實(shí)現(xiàn)PC端網(wǎng)頁(yè)微信掃碼支付(完整版)
這篇文章主要介紹了Java SpringMVC實(shí)現(xiàn)PC端網(wǎng)頁(yè)微信掃碼支付(完整版)的相關(guān)資料,非常不錯(cuò)具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2016-11-11
Java8利用Stream實(shí)現(xiàn)列表去重的方法詳解
這篇文章主要為大家介紹了Java利用Stream實(shí)現(xiàn)列表去重的幾種方法詳解,文中的示例代碼講解詳細(xì),需要的小伙伴可以參考一下2022-04-04
java 數(shù)據(jù)結(jié)構(gòu)之棧與隊(duì)列
這篇文章主要介紹了java 數(shù)據(jù)結(jié)構(gòu)之棧與隊(duì)列的相關(guān)資料,這里對(duì)java中的棧和隊(duì)列都做出實(shí)現(xiàn)實(shí)例來(lái)幫助大家理解學(xué)習(xí)數(shù)據(jù)結(jié)構(gòu),需要的朋友可以參考下2017-07-07
EntityWrapper如何在and條件中嵌套o(hù)r語(yǔ)句
這篇文章主要介紹了EntityWrapper如何在and條件中嵌套o(hù)r語(yǔ)句,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-03-03
Java中ThreadLocal避免內(nèi)存泄漏的方法詳解
ThreadLocal是Java中的一個(gè)線程本地存儲(chǔ)機(jī)制,它允許每個(gè)線程擁有一個(gè)獨(dú)立的本地存儲(chǔ)空間,用于存儲(chǔ)該線程的變量,本文主要介紹了ThreadLocal如何避免內(nèi)存泄漏,需要的朋友可以參考下2023-05-05

