這期內(nèi)容當(dāng)中小編將會給大家?guī)碛嘘P(guān)Java中為什么會出現(xiàn)Exception結(jié)構(gòu)異常,文章內(nèi)容豐富且以專業(yè)的角度為大家分析和敘述,閱讀完這篇文章希望大家可以有所收獲。
網(wǎng)站建設(shè)哪家好,找創(chuàng)新互聯(lián)!專注于網(wǎng)頁設(shè)計、網(wǎng)站建設(shè)、微信開發(fā)、成都微信小程序、集團企業(yè)網(wǎng)站建設(shè)等服務(wù)項目。為回饋新老客戶創(chuàng)新互聯(lián)還提供了天心免費建站歡迎大家使用!
1. Java異常Exception的結(jié)構(gòu)分析
我們通常所說的Exception主要是繼承于Throwable而來,可以參見如下的結(jié)構(gòu)圖示:
主要的Throwable分為異常和錯誤兩種,然后異常Exception和錯誤Error做為基類,分別被具體個性化以及衍生出NullPointerException、EOFException等等異常信息類。
基于Java中的源代碼來分析,Error和Exception僅僅是繼承了Throwable,做了構(gòu)造函數(shù)的拓展,沒有進行額外方法的延展;Exception輸出的主要核心方法都是定義在Throwable中的,感興趣的童鞋可以嘗試閱讀JDK的源代碼。
下面將介紹一下關(guān)鍵的幾個異常類方法:
1、getMessage(): String
輸出異常的描述信息
2、getLocalizedMessage()
輸出本地化的描述信息,一般此方法可被子類所覆蓋,缺省實現(xiàn)與getMessage()輸出信息一致
3、printStackTrace()
將異常棧打印到輸出流中,此為一類方法,默認打印到console控制臺,也可以顯式指定輸出流。
4、fillInStackTrace()
將當(dāng)前的異常棧保存到一個Throwable中,返回這個Throwable。大部分情況下,用在保留異常棧嵌套調(diào)用的情況,嘗試保留完整的異常棧,無需使用該方法。
2. Error vs Exception
Error在Java體系中定義為不可控制的問題,往往用來描述系統(tǒng)錯誤或者底層的問題,比如虛擬機錯誤,例如內(nèi)存空間不足,方法調(diào)用棧溢等。我們以上圖中列舉出的內(nèi)存溢出錯誤(StackOverflowError)為例,它是在JVM層面發(fā)生的錯誤,已經(jīng)游離于java應(yīng)用層之外;在應(yīng)用程序?qū)用媸菬o法進行捕獲,且無法從錯誤中恢復(fù)的。一般一旦發(fā)了類似問題,一般都是直接宕機,應(yīng)用停止正常的工作,需要重新啟動或者修復(fù)問題之后,方可重新正常工作。
Exception一般發(fā)生在應(yīng)用層,即在由項目中的Java代碼層面引發(fā)的問題,且可以嘗試進行捕獲,此類問題不會影響到應(yīng)用程序的正常工作的,即不會導(dǎo)致宕機現(xiàn)象的發(fā)生。我們在工作或者代碼中常見的都是Exception衍生出來的各類異常。
這里需要強調(diào)說明一下,JVM是Java語言的運行環(huán)境和平臺,但是并不是Java語言體系的一個部分;在JVM平臺上,還可以運行Groovy, JPython, JRuby, Closure,Scala等等遵守Java語言規(guī)范(JavaLanguage Specification)的編程語言,故我們可以將Error理解為脫離Java應(yīng)用之外的問題。
3. Exception中的運行時異常(RuntimeException)和受控異常(checked exception)
運行時異常(RuntimeException)是指在運行之時發(fā)生的異常,無需顯式地進行捕獲;如果程序中發(fā)生類似的異常,JVM會直接拋出此類異常,并打出響應(yīng)的異常棧信息。此類異常也通常被稱為unchecked exception, 未受控異常。
受控異常(checked Exception)是我們最常見的異常種類,在代碼中使用的異?;旧隙际谴祟惍惓?,此類異常會在代碼編譯階段由Java編譯器進行語法檢查,如果未顯式進行異常捕獲,則會報出相應(yīng)的編譯異常信息。
4. 如何在代碼中正確打印異常信息
下面我們將通過一系列的例子來說明上述幾個Exception中方法的使用技巧。
Case 1: getMessage()/getLocalizedMessage()
public void testCase1() { System.out.println("We are going to do something interesting...."); try { throw new NullPointerException("I am an exception in the code."); } catch (Exception e) { System.out.println("We got unexpected:" + e.getMessage()); System.out.println("We got unexpected:" + e.getLocalizedMessage()); } }
輸出結(jié)果:
We are going to do testing interesting.... We got unexpected in getMessage==> I am an exception in the code. We got unexpected in getLocalizedMessage==> I am an exception in the code.
基于結(jié)果來分析, 上述兩個方法只是將異常對象中的Message打印出來,這些信息對于我們追蹤問題和調(diào)試幫助有限。
Case 2:e.printStackTrace()
public void testCase2() { System.out.println("We are going to do something interesting...."); try { throw new Exception("I am an exception in the code."); } catch (Exception e) { e.printStackTrace(); } }
運行結(jié)果:
運行結(jié)果圖
printStackTrace()
可以打印出整個異常棧,但是異常棧信息將輸出到默認的輸出流中,絕大多數(shù)情況下是系統(tǒng)的控制臺,而在實際項目中,都是需要將異常棧輸出到日志文件的,如果不顯式指定,則會丟失異常信息,在日志文件中無從追查。
Case 3: 基于log4j/slf4j等輸出到日志文件
在實際項目中,一般會使用log4j/log4j2/JDK logging/slf4j/logback等日志系統(tǒng)來存放日志,那如何來講日志的異常棧存入日志文件呢,我們來看示例。
public void testCase3() { System.out.println("We are going to do something interesting...."); try { throw new NullPointerException("abcedfeg"); } catch (Exception e) { logger.info("Unexpected error in ", e); } }
我們需要到日志文件中,找到相應(yīng)的異常信息,異常信息如下:
12:24:45.387 [main] INFO org.demo.TestException - Unexpected error in java.lang.NullPointerException: abcedfeg at org.demo.TestException.testCase3(TestException.java:39) at org.demo.TestException.main(TestException.java:12)
我們可以發(fā)現(xiàn),整個異常棧信息由兩個部分組成:
>> 異常中的message, 類似getMessage()
輸出的信息,
>> 使用logger.info
之類的方法,將異常信息寫入到日志流中.
以下為log4j的聲明,這里以slf4j為例來示例:
private static final Logger logger = LoggerFactory.getLogger(TestException.class);
Case 4: fillInStackTrace()
public class FillInExceptionTest { public static void main(String[] args) { FillInExceptionTest fit = new FillInExceptionTest(); try { fit.outerMehtod(); } catch (Exception e) { System.out.println("\n==========I am the one evil separation line=============="); e.printStackTrace(); } } public void innerMethod() throws Exception { throw new Exception("I got exception in an inner method."); } public void outerMehtod() throws Exception { try { innerMethod(); //invoke inner method. } catch (Exception e) { e.printStackTrace(); throw (Exception)e.fillInStackTrace(); } } }
運行結(jié)果:
基于上述的運行結(jié)果可知,fillInStackTrace()
只提取了當(dāng)下的異常棧信息,而非完整的異常棧信息,這個就是此方法帶給我們的特殊之處。
如果我們需要在最外層將完整的異常棧打印出來,該如何做呢? 將下述的語句:
throw (Exception)e.fillInStackTrace();
替換為:
throw e;
重新運行程序,我們就可以在最外層得到完整的異常棧信息了。
上述就是小編為大家分享的Java中為什么會出現(xiàn)Exception結(jié)構(gòu)異常了,如果剛好有類似的疑惑,不妨參照上述分析進行理解。如果想知道更多相關(guān)知識,歡迎關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道。