java 異常機制_深入理解Java異常處理機制

一、引子

try…catch…finally恐怕是大家再熟悉不過的語句了,而且感覺用起來也是很簡單,邏輯上似乎也是很容易理解。不過,我親自體驗的“教訓”告訴我,這個東西可不是想象中的那么簡單、聽話。不信?那你看看下面的代碼,“猜猜”它執行后的結果會是什么?不要往后看答案、也不許執行代碼看真正答案哦。如果你的答案是正確,那么這篇文章你就不用浪費時間看啦。

1 packageexception;2

3 /**

4 *@authorzsh5 * @company wlgzs6 * @create 2019-03-01 10:077 * @Describe 異常處理測試8 */

9 public classTestException {10

11 boolean testEx() throwsException {12 boolean ret = true;13 try{14 ret =testEx1();15 } catch(Exception e) {16 System.out.println("testEx, catch exception");17 ret = false;18 throwe;19 } finally{20 System.out.println("testEx, finally; return value=" +ret);21 returnret;22 }23 }24 boolean testEx1() throwsException {25 boolean ret = true;26 try{27 ret =testEx2();28 if (!ret) {29 return false;30 }31 System.out.println("testEx1, at the end of try");32 returnret;33 } catch(Exception e) {34 System.out.println("testEx1, catch exception");35 ret = false;36 throwe;37 } finally{38 System.out.println("testEx1, finally; return value=" +ret);39 returnret;40 }41 }42 boolean testEx2() throwsException {43 boolean ret = true;44 try{45 int b = 12;46 intc;47 for (int i = 2; i >= -2; i--) {48 c = b /i;49 System.out.println("i=" +i);50 }51 return true;52 } catch(Exception e) {53 System.out.println("testEx2, catch exception");54 ret = false;55 throwe;56 } finally{57 System.out.println("testEx2, finally; return value=" +ret);58 returnret;59 }60 }61 public static voidmain(String[] args) {62 TestException testException1 = newTestException();63 try{64 testException1.testEx();65 } catch(Exception e) {66 e.printStackTrace();67 }68 }69 }

你的答案是什么?是下面的答案嗎?

i=2

i=1

testEx2, catch exception

testEx2, finally; return value=false

testEx1, catch exception

testEx1, finally; return value=false

testEx, catch exception

testEx, finally; return value=false

如果你的答案真的如上面所說,那么你錯啦。^_^,那就建議你仔細看一看這篇文章或者拿上面的代碼按各種不同的情況修改、執行、測試,你會發現有很多事情不是原來想象中的那么簡單的。現在公布正確答案:

d4d09c041259a62a633acaa20930e600.png

注意說明:finally語句塊不應該出現 應該出現return。上面的return ret最好是其他語句來處理相關邏輯。

二、JAVA異常

異常指不期而至的各種狀況,如:文件找不到、網絡連接失敗、非法參數等。異常是一個事件,它發生在程序運行期間,干擾了正常的指令流程。Java通 過API中Throwable類的眾多子類描述各種不同的異常。因而,Java異常都是對象,是Throwable子類的實例,描述了出現在一段編碼中的 錯誤條件。當條件生成時,錯誤將引發異常。

Java異常類層次結構圖:

e3cb2937cdaf6a386bbc1f9cac30bd8a.png

在 Java 中,所有的異常都有一個共同的祖先 Throwable(可拋出)。Throwable 指定代碼中可用異常傳播機制通過 Java 應用程序傳輸的任何問題的共性。

Throwable: 有兩個重要的子類:Exception(異常)和 Error(錯誤),二者都是 Java 異常處理的重要子類,各自都包含大量子類。

Error(錯誤):是程序無法處理的錯誤,表示運行應用程序中較嚴重問題。大多數錯誤與代碼編寫者執行的操作無關,而表示代碼運行時 JVM(Java 虛擬機)出現的問題。例如,Java虛擬機運行錯誤(Virtual MachineError),當 JVM 不再有繼續執行操作所需的內存資源時,將出現 OutOfMemoryError。這些異常發生時,Java虛擬機(JVM)一般會選擇線程終止。

這些錯誤表示故障發生于虛擬機自身、或者發生在虛擬機試圖執行應用時,如Java虛擬機運行錯誤(Virtual MachineError)、類定義錯誤(NoClassDefFoundError)等。這些錯誤是不可查的,因為它們在應用程序的控制和處理能力之 外,而且絕大多數是程序運行時不允許出現的狀況。對于設計合理的應用程序來說,即使確實發生了錯誤,本質上也不應該試圖去處理它所引起的異常狀況。在 Java中,錯誤通過Error的子類描述。

Exception(異常):是程序本身可以處理的異常。

Exception 類有一個重要的子類 RuntimeException。RuntimeException 類及其子類表示“JVM 常用操作”引發的錯誤。例如,若試圖使用空值對象引用、除數為零或數組越界,則分別引發運行時異常(NullPointerException、ArithmeticException)和 ArrayIndexOutOfBoundException。

注意:異常和錯誤的區別:異常能被程序本身可以處理,錯誤是無法處理。

通常,Java的異常(包括Exception和Error)分為可查的異常(checked exceptions)和不可查的異常(unchecked exceptions)。

可查異常(編譯器要求必須處置的異常):正確的程序在運行中,很容易出現的、情理可容的異常狀況。可查異常雖然是異常狀況,但在一定程度上它的發生是可以預計的,而且一旦發生這種異常狀況,就必須采取某種方式進行處理。

除了RuntimeException及其子類以外,其他的Exception類及其子類都屬于可查異常。這種異常的特點是Java編譯器會檢查它,也就是說,當程序中可能出現這類異常,要么用try-catch語句捕獲它,要么用throws子句聲明拋出它,否則編譯不會通過。

不可查異常(編譯器不要求強制處置的異常):包括運行時異常(RuntimeException與其子類)和錯誤(Error)。

Exception 這種異常分兩大類運行時異常和非運行時異常(編譯異常)。程序中應當盡可能去處理這些異常。

運行時異常:都是RuntimeException類及其子類異常,如NullPointerException(空指針異常)、IndexOutOfBoundsException(下標越界異常)等,這些異常是不檢查異常,程序中可以選擇捕獲處理,也可以不處理。這些異常一般是由程序邏輯錯誤引起的,程序應該從邏輯角度盡可能避免這類異常的發生。

運行時異常的特點是Java編譯器不會檢查它,也就是說,當程序中可能出現這類異常,即使沒有用try-catch語句捕獲它,也沒有用throws子句聲明拋出它,也會編譯通過。

非運行時異常?(編譯異常):是RuntimeException以外的異常,類型上都屬于Exception類及其子類。從程序語法角度講是必須進行處理的異常,如果不處理,程序就不能編譯通過。如IOException、SQLException等以及用戶自定義的Exception異常,一般情況下不自定義檢查異常。

三、Java處理異常機制

在 Java 應用程序中,異常處理機制為:拋出異常,捕捉異常。

拋出異常:當一個方法出現錯誤引發異常時,方法創建異常對象并交付運行時系統,異常對象中包含了異常類型和異常出現時的程序狀態等異常信息。運行時系統負責尋找處置異常的代碼并執行。

捕獲異常:在方法拋出異常之后,運行時系統將轉為尋找合適的異常處理器(exception handler)。潛在的異常處理器是異常發生時依次存留在調用棧中的方法的集合。當異常處理器所能處理的異常類型與方法拋出的異常類型相符時,即為合適 的異常處理器。運行時系統從發生異常的方法開始,依次回查調用棧中的方法,直至找到含有合適異常處理器的方法并執行。當運行時系統遍歷調用棧而未找到合適 的異常處理器,則運行時系統終止。同時,意味著Java程序的終止。

對于運行時異常、錯誤或可查異常,Java技術所要求的異常處理方式有所不同。

由于運行時異常的不可查性,為了更合理、更容易地實現應用程序,Java規定,運行時異常將由Java運行時系統自動拋出,允許應用程序忽略運行時異常。

對于方法運行中可能出現的Error,當運行方法不欲捕捉時,Java允許該方法不做任何拋出聲明。因為,大多數Error異常屬于永遠不能被允許發生的狀況,也屬于合理的應用程序不該捕捉的異常。

對于所有的可查異常,Java規定:一個方法必須捕捉,或者聲明拋出方法之外。也就是說,當一個方法選擇不捕捉可查異常時,它必須聲明將拋出異常。

能夠捕捉異常的方法,需要提供相符類型的異常處理器。所捕捉的異常,可能是由于自身語句所引發并拋出的異常,也可能是由某個調用的方法或者Java運行時 系統等拋出的異常。也就是說,一個方法所能捕捉的異常,一定是Java代碼在某處所拋出的異常。簡單地說,異常總是先被拋出,后被捕捉的。

任何Java代碼都可以拋出異常,如:自己編寫的代碼、來自Java開發環境包中代碼,或者Java運行時系統。無論是誰,都可以通過Java的throw語句拋出異常。

從方法中拋出的任何異常都必須使用throws子句。

捕捉異常通過try-catch語句或者try-catch-finally語句實現。

總體來說,Java規定:對于可查異常必須捕捉、或者聲明拋出。允許忽略不可查的RuntimeException和Error。

3.1?捕獲異常:try、catch 和 finally

3.1.1 try-catch語句

在Java中,異常通過try-catch語句捕獲。其一般語法形式為:

1 try{2 //可能會發生異常的程序代碼

3 } catch(Type1 id1){4 //捕獲并處置try拋出的異常類型Type1

5 }6 catch(Type2 id2){7 //捕獲并處置try拋出的異常類型Type2

8 }

關鍵詞try后的一對大括號將一塊可能發生異常的代碼包起來,稱為監控區域。Java方法在運行過程中出現異常,則創建異常對象。將異常拋出監控區域之 外,由Java運行時系統試圖尋找匹配的catch子句以捕獲異常。若有匹配的catch子句,則運行其異常處理代碼,try-catch語句結束。

匹配的原則是:如果拋出的異常對象屬于catch子句的異常類,或者屬于該異常類的子類,則認為生成的異常對象與catch塊捕獲的異常類型相匹配。

例1? 捕捉throw語句拋出的“除數為0”異常。

1 /**

2 * 捕捉throw語句拋出的“除數為0”異常。3 */

4 static voidf1(){5 int a = 6;6 int b = 0;7 //try監控區域

8 try{9 //通過throw語句拋出異常

10 if (b == 0) throw newArithmeticException();11 System.out.println("a/b的值是:" + a /b);12 }13 //catch捕捉異常

14 catch(ArithmeticException e) {15 System.out.println("程序出現異常,變量b不能為0。");16 }17 System.out.println("程序正常結束。");18 }

運行結果:

b37feba8c98e1266d8c7691cb089fb0d.png

例1? 在try監控區域通過if語句進行判斷,當“除數為0”的錯誤條件成立時引發ArithmeticException異常,創建 ArithmeticException異常對象,并由throw語句將異常拋給Java運行時系統,由系統尋找匹配的異常處理器catch并運行相應異 常處理代碼,打印輸出“程序出現異常,變量b不能為0。”try-catch語句結束,繼續程序流程。

事實上,“除數為0”等ArithmeticException,是RuntimException的子類。而運行時異常將由運行時系統自動拋出,不需要使用throw語句。

例2??捕捉運行時系統自動拋出“除數為0”引發的ArithmeticException異常。

1 /**

2 * 捕捉運行時系統自動拋出“除數為0”引發的ArithmeticException異常。3 */

4 static voidf2(){5 int a = 6;6 int b = 0;7 try{8 System.out.println("a/b的值是:" + a /b);9 } catch(ArithmeticException e) {10 System.out.println("程序出現異常,變量b不能為0。");11 }12 System.out.println("程序正常結束。");13 }

運行結果:

315aaedd742b30c005dd7e5c4a509312.png

例2? 中的語句:System.out.println("a/b的值是:" + a/b);

在運行中出現“除數為0”錯誤,引發ArithmeticException異常。運行時系統創建異常對象并拋出監控區域,轉而匹配合適的異常處理器catch,并執行相應的異常處理代碼。

由于檢查運行時異常的代價遠大于捕捉異常所帶來的益處,運行時異常不可查。Java編譯器允許忽略運行時異常,一個方法可以既不捕捉,也不聲明拋出運行時異常。

例3??不捕捉、也不聲明拋出運行時異常。

1 /**

2 * 不捕捉、也不聲明拋出運行時異常。3 */

4 static voidf3(){5 inta, b;6 a = 6;7 b = 0; //除數b 的值為0

8 System.out.println(a /b);9 }

運行結果:

5d7215c4f514c553601c25eb48e64710.png

例4? 程序可能存在除數為0異常和數組下標越界異常。

1 /**

2 * 程序可能存在除數為0異常和數組下標越界異常。3 */

4 static voidf4(){5 int[] intArray = new int[3];6 try{7 for (int i = 0; i <= intArray.length; i++) {8 intArray[i] =i;9 System.out.println("intArray[" + i + "] = " +intArray[i]);10 System.out.println("intArray[" + i + "]模 " + (i - 2) + "的值: "

11 + intArray[i] % (i - 2));12 }13 } catch(ArrayIndexOutOfBoundsException e) {14 System.out.println("intArray數組下標越界異常。");15 } catch(ArithmeticException e) {16 System.out.println("除數為0異常。");17 }18 System.out.println("程序正常結束。");19 }

運行結果:

ce13ed1ad170d72458f98c229340bc72.png

例4? 程序可能會出現除數為0異常,還可能會出現數組下標越界異常。程序運行過程中ArithmeticException異常類型是先行匹配的,因此執行相匹配的catch語句:

1 catch(ArithmeticException e){2 System.out.println("除數為0異常。");3 }

需要注意的是,一旦某個catch捕獲到匹配的異常類型,將進入異常處理代碼。一經處理結束,就意味著整個try-catch語句結束。其他的catch子句不再有匹配和捕獲異常類型的機會。

Java通過異常類描述異常類型,異常類的層次結構如圖1所示。對于有多個catch子句的異常程序而言,應該盡量將捕獲底層異常類的catch子 句放在前面,同時盡量將捕獲相對高層的異常類的catch子句放在后面。否則,捕獲底層異常類的catch子句將可能會被屏蔽。

RuntimeException異常類包括運行時各種常見的異常,ArithmeticException類和ArrayIndexOutOfBoundsException類都是它的子類。因此,RuntimeException異常類的catch子句應該放在 最后面,否則可能會屏蔽其后的特定異常處理或引起編譯錯誤。

3.1.2 try-catch-finally語句

try-catch語句還可以包括第三部分,就是finally子句。它表示無論是否出現異常,都應當執行的內容。try-catch-finally語句的一般語法形式為:

1 try{2 //可能會發生異常的程序代碼

3 } catch(Type1 id1) {4 //捕獲并處理try拋出的異常類型Type1

5 } catch(Type2 id2) {6 //捕獲并處理try拋出的異常類型Type2

7 } finally{8 }

例5??帶finally子句的異常處理程序。

1 /**

2 * 帶finally子句的異常處理程序。3 */

4 static voidf5(){5 int i = 0;6 String greetings[] = { " Hello world !", " Hello World !! ",7 " HELLO WORLD !!!"};8 while (i < 4) {9 try{10 //特別注意循環控制變量i的設計,避免造成無限循環

11 System.out.println(greetings[i++]);12 } catch(ArrayIndexOutOfBoundsException e) {13 System.out.println("數組下標越界異常");14 } finally{15 System.out.println("--------------------------");16 }17 }18 }

運行結果:

3dbcfd2eccdc1794c7813be32717796a.png

在例5中,請特別注意try子句中語句塊的設計,如果設計為如下,將會出現死循環。如果設計為:

1 try{2 System.out.println (greetings[i]); i++;3 }

小結:

try 塊:用于捕獲異常。其后可接零個或多個catch塊,如果沒有catch塊,則必須跟一個finally塊。

catch 塊:用于處理try捕獲到的異常。

finally 塊:無論是否捕獲或處理異常,finally塊里的語句都會被執行。當在try塊或catch塊中遇到return語句時,finally語句塊將在方法返回之前被執行。在以下4種特殊情況下,finally塊不會被執行:

1)在finally語句塊中發生了異常。

2)在前面的代碼中用了System.exit()退出程序。

3)程序所在的線程死亡。

4)關閉CPU。

3.1.3?try-catch-finally 規則(異常處理語句的語法規則):

1) ?必須在 try 之后添加 catch 或 finally 塊。try 塊后可同時接 catch 和 finally 塊,但至少有一個塊。

2) 必須遵循塊順序:若代碼同時使用 catch 和 finally 塊,則必須將 catch 塊放在 try 塊之后。

3) catch 塊與相應的異常類的類型相關。

4) 一個 try 塊可能有多個 catch 塊。若如此,則執行第一個匹配塊。即Java虛擬機會把實際拋出的異常對象依次和各個catch代碼塊聲明的異常類型匹配,如果異常對象為某個異常類型或其子類的實例,就執行這個catch代碼塊,不會再執行其他的 catch代碼塊

5) 可嵌套 try-catch-finally 結構。

6) 在 try-catch-finally 結構中,可重新拋出異常。

7) 除了下列情況,總將執行 finally 做為結束:JVM 過早終止(調用 System.exit(int));在 finally 塊中拋出一個未處理的異常;計算機斷電、失火、或遭遇病毒攻擊。

1)當try沒有捕獲到異常時:try語句塊中的語句逐一被執行,程序將跳過catch語句塊,執行finally語句塊和其后的語句;

2)當try捕獲到異常,catch語句塊里沒有處理此異常的情況:當try語句塊里的某條語句出現異常時,而沒有處理此異常的catch語句塊時,此異常將會拋給JVM處理,finally語句塊里的語句還是會被執行,但finally語句塊后的語句不會被執行;

3)當try捕獲到異常,catch語句塊里有處理此異常的情況:在try語句塊中是按照順序來執行的,當執行到某一條語句出現異常時,程序將跳到catch語句塊,并與catch語句塊逐一匹配,找到與之對應的處理程序,其他的catch語句塊將不會被執行,而try語句塊中,出現異常之后的語句也不會被執行,catch語句塊執行完后,執行finally語句塊里的語句,最后執行finally語句塊后的語句;

圖示try、catch、finally語句塊的執行:

157dfc47bfdd61cd23b1123ff7fc15f4.png

3.2 拋出異常

任何Java代碼都可以拋出異常,如:自己編寫的代碼、來自Java開發環境包中代碼,或者Java運行時系統。無論是誰,都可以通過Java的throw語句拋出異常。從方法中拋出的任何異常都必須使用throws子句。

3.2.1 throws拋出異常

如果一個方法可能會出現異常,但沒有能力處理這種異常,可以在方法聲明處用throws子句來聲明拋出異常。例如汽車在運行時可能會出現故障,汽車本身沒辦法處理這個故障,那就讓開車的人來處理。

throws語句用在方法定義時聲明該方法要拋出的異常類型,如果拋出的是Exception異常類型,則該方法被聲明為拋出所有的異常。多個異常可使用逗號分割。throws語句的語法格式為:

1 methodname throwsException1,Exception2,..,ExceptionN {2 }

方法名后的throws Exception1,Exception2,...,ExceptionN 為聲明要拋出的異常列表。當方法拋出異常列表的異常時,方法將不對這些類型及其子類類型的異常作處理,而拋向調用該方法的方法,由他去處理。例如:

1 packageexception;2

3 /**

4 *@authorzsh5 * @company wlgzs6 * @create 2019-03-01 21:307 * @Describe 異常處理測試8 */

9 public classMain3 {10

11 static void pop() throwsNegativeArraySizeException {12 //定義方法并拋出NegativeArraySizeException異常

13 int[] arr = new int[-3]; //創建數組

14 }15

16 public static voidmain(String[] args) {17 try { //try語句處理異常信息

18 pop(); //調用pop()方法

19 } catch(NegativeArraySizeException e) {20 System.out.println("pop()方法拋出的異常");//輸出異常信息

21 }22 }23 }

運行結果:

aaee6de54d6ee65099f5411c39602ef5.png

使用throws關鍵字將異常拋給調用者后,如果調用者不想處理該異常,可以繼續向上拋出,但最終要有能夠處理該異常的調用者。pop方法沒有處理異常NegativeArraySizeException,而是由main函數來處理。

Throws拋出異常的規則:

1) 如果是不可查異常(unchecked exception),即Error、RuntimeException或它們的子類,那么可以不使用throws關鍵字來聲明要拋出的異常,編譯仍能順利通過,但在運行時會被系統拋出。

2)必須聲明方法可拋出的任何可查異常(checked exception)。即如果一個方法可能出現受可查異常,要么用try-catch語句捕獲,要么用throws子句聲明將它拋出,否則會導致編譯錯誤

3)僅當拋出了異常,該方法的調用者才必須處理或者重新拋出該異常。當方法的調用者無力處理該異常的時候,應該繼續拋出,而不是囫圇吞棗。

4)調用方法必須遵循任何可查異常的處理和聲明規則。若覆蓋一個方法,則不能聲明與覆蓋方法不同的異常。聲明的任何異常必須是被覆蓋方法所聲明異常的同類或子類。

例如:

1 //合法

2 void method1() throwsIOException{}3

4 //編譯錯誤,必須捕獲或聲明拋出IOException

5 voidmethod2(){6 method1();7 }8

9 //合法,聲明拋出IOException

10 void method3()throwsIOException {11 method1();12 }13

14 //合法,聲明拋出Exception,IOException是Exception的子類

15 void method4()throwsException {16 method1();17 }18

19 //合法,捕獲IOException

20 voidmethod5(){21 try{22 method1();23 }catch(IOException e){ }24 }25

26 //編譯錯誤,必須捕獲或聲明拋出Exception

27 voidmethod6(){28 try{29 method1();30 }catch(IOException e){throw newException();}31 }32

33 //合法,聲明拋出Exception

34 void method7() throwsException{35 try{36 method1();37 }catch(IOException e){throw newException();}38 }

判斷一個方法可能會出現異常的依據如下:

1)方法中有throw語句。例如,以上method7()方法的catch代碼塊有throw語句。  2)調用了其他方法,其他方法用throws子句聲明拋出某種異常。例如,method3()方法調用了method1()方法,method1()方法聲明拋出IOException,因此,在method3()方法中可能會出現IOException。

3.2.2?使用throw拋出異常

throw總是出現在函數體中,用來拋出一個Throwable類型的異常。程序會在throw語句后立即終止,它后面的語句執行不到,然后在包含它的所有try塊中(可能在上層調用函數中)從里向外尋找含有與其匹配的catch子句的try塊。

我們知道,異常是異常類的實例對象,我們可以創建異常類的實例對象通過throw語句拋出。該語句的語法格式為:

throw new exceptionname;

例如拋出一個IOException類的異常對象:

throw new IOException;

要注意的是,throw 拋出的只能夠是可拋出類Throwable 或者其子類的實例對象。下面的操作是錯誤的:

throw new String("exception");

這是因為String 不是Throwable 類的子類。

如果拋出了檢查異常,則還應該在方法頭部聲明方法可能拋出的異常類型。該方法的調用者也必須檢查處理拋出的異常。

如果所有方法都層層上拋獲取的異常,最終JVM會進行處理,處理也很簡單,就是打印異常消息和堆棧信息。如果拋出的是Error或RuntimeException,則該方法的調用者可選擇處理該異常。

1 packageexception;2

3 /**

4 *@authorzsh5 * @company wlgzs6 * @create 2019-03-01 21:427 * @Describe 使用throw拋出異常8 */

9 public classMain4 {10

11 static int quotient(int x, int y) throws MyException { //定義方法拋出異常

12 if (y < 0) { //判斷參數是否小于0

13 throw new MyException("除數不能是負數"); //異常信息

14 }15 return x/y; //返回值

16 }17 public static void main(String args[]) { //主方法

18 int a =3;19 int b = -1;20 try { //try語句包含可能發生異常的語句

21 int result = quotient(a, b); //調用方法quotient()

22 } catch (MyException e) { //處理自定義異常

23 System.out.println(e.getMessage()); //輸出異常信息

24 } catch (ArithmeticException e) { //處理ArithmeticException異常

25 System.out.println("除數不能為0"); //輸出提示信息

26 } catch (Exception e) { //處理其他異常

27 System.out.println("程序發生了其他的異常"); //輸出提示信息

28 }29 }30 }31

32 class MyException extends Exception { //創建自定義異常類

33 String message; //定義String類型變量

34 public MyException(String ErrorMessagr) { //父類方法

35 message =ErrorMessagr;36 }37

38 public String getMessage() { //覆蓋getMessage()方法

39 returnmessage;40 }41 }

運行結果:

fcb592817b6b8d619d469c8b32d1562d.png

3.3 異常鏈

1) 如果調用quotient(3,-1),將發生MyException異常,程序調轉到catch (MyException e)代碼塊中執行;

2) 如果調用quotient(5,0),將會因“除數為0”錯誤引發ArithmeticException異常,屬于運行時異常類,由Java運行時系統自動拋出。quotient()方法沒有捕捉ArithmeticException異常,Java運行時系統將沿方法調用棧查到main方法,將拋出的異常上傳至quotient()方法的調用者:

int result = quotient(a, b); // 調用方法quotient()

由于該語句在try監控區域內,因此傳回的“除數為0”的ArithmeticException異常由Java運行時系統拋出,并匹配catch子句:

catch (ArithmeticException e) { // 處理ArithmeticException異常

System.out.println("除數不能為0"); // 輸出提示信息

}

處理結果是輸出“除數不能為0”。Java這種向上傳遞異常信息的處理機制,形成異常鏈。

Java方法拋出的可查異常將依據調用棧、沿著方法調用的層次結構一直傳遞到具備處理能力的調用方法,最高層次到main方法為止。如果異常傳遞到main方法,而main不具備處理能力,也沒有通過throws聲明拋出該異常,將可能出現編譯錯誤。

3)如還有其他異常發生,將使用catch (Exception e)捕捉異常。由于Exception是所有異常類的父類,如果將catch (Exception e)代碼塊放在其他兩個代碼塊的前面,后面的代碼塊將永遠得不到執行,就沒有什么意義了,所以catch語句的順序不可掉換。

3.4?Throwable類中的常用方法

注意:catch關鍵字后面括號中的Exception類型的參數e。Exception就是try代碼塊傳遞給catch代碼塊的變量類型,e就是變量名。catch代碼塊中語句"e.getMessage();"用于輸出錯誤性質。通常異常處理常用3個函數來獲取異常的有關信息:

getCause():返回拋出異常的原因。如果 cause 不存在或未知,則返回 null。

getMeage():返回異常的消息信息。

printStackTrace():對象的堆棧跟蹤輸出至錯誤輸出流,作為字段 System.err 的值。

有時為了簡單會忽略掉catch語句后的代碼,這樣try-catch語句就成了一種擺設,一旦程序在運行過程中出現了異常,就會忽略處理異常,而錯誤發生的原因很難查找。

四、Java常見異常

在Java中提供了一些異常用來描述經常發生的錯誤,對于這些異常,有的需要程序員進行捕獲處理或聲明拋出,有的是由Java虛擬機自動進行捕獲處理。Java中常見的異常類:

4.1?runtimeException子類:

1、java.lang.ArrayIndexOutOfBoundsException

數組索引越界異常。當對數組的索引值為負數或大于等于數組大小時拋出。2、java.lang.ArithmeticException

算術條件異常。譬如:整數除零等。3、java.lang.NullPointerException

空指針異常。當應用試圖在要求使用對象的地方使用了null時,拋出該異常。譬如:調用null對象的實例方法、訪問null對象的屬性、計算null對象的長度、使用throw語句拋出null等等4、java.lang.ClassNotFoundException

找不到類異常。當應用試圖根據字符串形式的類名構造類,而在遍歷CLASSPAH之后找不到對應名稱的class文件時,拋出該異常。5、java.lang.NegativeArraySizeException

數組長度為負異常6、java.lang.ArrayStoreException

數組中包含不兼容的值拋出的異常7、java.lang.SecurityException

安全性異常8、java.lang.IllegalArgumentException

非法參數異常

4.2?IOException

IOException:操作輸入流和輸出流時可能出現的異常。

EOFException:文件已結束異常

FileNotFoundException:文件未找到異常

4.3 其他

ClassCastException ? ?類型轉換異常類

ArrayStoreException ?數組中包含不兼容的值拋出的異常

SQLException ? 操作數據庫異常類

NoSuchFieldException ? 字段未找到異常

NoSuchMethodException ? 方法未找到拋出的異常

NumberFormatException ? ?字符串轉換為數字拋出的異常

StringIndexOutOfBoundsException 字符串索引超出范圍拋出的異常

IllegalAccessException ?不允許訪問某類異常

InstantiationException ?當應用程序試圖使用Class類中的newInstance()方法創建一個類的實例,而指定的類對象無法被實例化時,拋出該異常

五、自定義異常

使用Java內置的異常類可以描述在編程時出現的大部分異常情況。除此之外,用戶還可以自定義異常。用戶自定義異常類,只需繼承Exception類即可。

在程序中使用自定義異常類,大體可分為以下幾個步驟。

(1)創建自定義異常類。

(2)在方法中通過throw關鍵字拋出異常對象。

(3)如果在當前拋出異常的方法中處理異常,可以使用try-catch語句捕獲并處理;否則在方法的聲明處通過throws關鍵字指明要拋出給方法調用者的異常,繼續進行下一步操作。

(4)在出現異常方法的調用者中捕獲并處理異常。

在上面的“使用throw拋出異常”例子已經提到了。

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/272856.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/272856.shtml
英文地址,請注明出處:http://en.pswp.cn/news/272856.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

ThreadLocal http://blog.jobbole.com/20400/

d轉載于:https://www.cnblogs.com/hansongjiang/p/4875332.html

做人:失信是最大的破產!

一個人最大的破產是信用的破產&#xff01;哪怕你一無所有&#xff0c;但只要信用還在&#xff0c;就還有翻身的本金。保護好信用&#xff0c;珍惜別人給你的每一次信任&#xff01;因為有時候我們只有一次機會&#xff01;朋友有時候就像鈔票&#xff0c;有真也有假。我們需要…

c#和WIN32 調用

作者&#xff1a;劉鐵猛日期&#xff1a;2005-12-20關鍵字&#xff1a;C# .NET Win32 API版權聲明:本文章受知識產權法保護&#xff0c;如果閣下想轉載,在轉載的時候煩勞閣下連同在下的姓名一起轉載,并向bladeytom.com發一個Mail,我很想知道我的文章都去哪里了.謝謝.小序Win32 …

【干貨】微信公眾號運營必備工具(完整版)

做微信公眾號運營最基本的要素有兩點&#xff1a;一是內容要強大&#xff08;內&#xff09;&#xff0c;二是排版要美觀&#xff08;外&#xff09;。做好前者&#xff0c;你需要有充足的知識儲備以及精彩獨到的文筆作為支撐&#xff0c;而做好后者則相對簡單許多&#xff0c;…

java定時線程池_java 定時器線程池(ScheduledThreadPoolExecutor)的實現

前言定時器線程池提供了定時執行任務的能力&#xff0c;即可以延遲執行&#xff0c;可以周期性執行。但定時器線程池也還是線程池&#xff0c;最底層實現還是ThreadPoolExecutor&#xff0c;可以參考我的另外一篇文章多線程–精通ThreadPoolExecutor。特點說明1.構造函數public…

iOS 關于關鍵字高亮

- (NSMutableAttributedString *)colorStr: (NSString *)originalStr // originalStr : 需要高亮傳入的字符串 { NSMutableAttributedString *dataStr [[[NSMutableAttributedString alloc] initWithString:originalStr] autorelease]; for (int i 0; i < originalStr.l…

成功,要“借力”,不要“盡力”(深刻!)

01每個人都喜歡成功&#xff0c;卻又時常感覺自己力不從心一個小男孩在院子里搬一塊石頭&#xff0c;父親在旁邊鼓勵&#xff1a;“孩子&#xff0c;只要你全力以赴&#xff0c;一定搬得起來&#xff01;”但是石頭太重&#xff0c;最終孩子也沒能搬起來。他告訴父親&#xff1…

java 網站開發實例_完整的javaweb項目

【實例簡介】主要功能有以下幾個&#xff1a;1.用戶注冊 2.用戶登錄 3.用戶列表展示 4.用戶信息修改 5.用戶信息刪除【實例截圖】【核心代碼】javaweb└── javaweb├── src│ └── com│ ├── dao│ │ ├── UserDaoImpl.java│ │ └── UserDao.java│…

Nginx+Php-fpm+MySQL+Redis源碼編譯安裝指南

說明&#xff1a;本教程由三部分組成如下&#xff1a; 1. 源碼編譯安裝Nginx 2. 源碼編譯安裝php以及mysql、redis擴展模塊 3. 配置虛擬主機 文中所涉及安裝包程序均提供下載鏈接&#xff0c;歡迎使用 執行環境以及前置條件&#xff1a;Ubuntu 12.04 LTS 已安裝…

NetFramework各個版本的特性筆記

我的博客&#xff1a;http://www.cnblogs.com/hgmyz/p/6916064.html公式記憶&#xff1a;.Net 2.0CLRBCLC#(VB.Net)Win FormWeb Form.Net 3.0.Net 2.0WCFWPFWFWCS.Net 3.5.Net 3.0Asp.Net AjaxSliverlightLinq.Net 4.0 增加了并行的支持&#xff0c;與舊的Framwork并行工作。默…

從0開始學Java——JSPServlet——HttpServletRequest相關的幾個路徑信息

在HttpServletRequest中有幾個獲取路徑的接口&#xff1a;getRequestURI/getContextPath/getServletPath/getPathInfo 這些接口互相之間有什么區別&#xff0c;通過下面這段代碼就可以分辨清楚了&#xff1a; 1 WebServlet("/hello.view")2 public class FirstServle…

C#編譯和運行過程圖例

一張圖&#xff0c;描述C#編譯和運行過程&#xff0c;比較容易記憶理解

java 不重啟部署_編譯Java類后不重啟Tomcat有兩種方式:熱部署、熱加載

不重啟Tomcat有兩種方式&#xff1a;熱部署、熱加載熱部署&#xff1a;容器狀況在運行時重新部署整個項目。這類環境下一般整個內存會清空,重新加載&#xff0c;這類方式有可能會造成sessin丟失等環境。tomcat 6確實可以熱部署了,而且對話也沒丟.熱加載&#xff1a;最好是在調試…

修改mysql的用戶密碼

修改的用戶都以root為列。一、擁有原來的myql的root的密碼&#xff1b; 方法一&#xff1a; #mysql -u root mysql> SET PASSWORD FOR rootlocalhost PASSWORD(newpass); 方法二&#xff1a;在mysql系統外&#xff0c;使用mysqladmin# mysqladmin -u root -p password &quo…

C#中的堆和棧理解

引言&#xff1a;程序運行時&#xff0c;它的數據必須存在內存中&#xff0c;一個數據需要多大內存、存儲在什么地方以及如何存儲都依賴于該數據的數據類型。1、什么是棧棧是一個內存數組&#xff0c;是一個LIFO&#xff08;Last-In-First-Out 后進先出&#xff09;的數據結構。…

java sessionmanager_java.lang.IllegalStateException:沒有SessionManager

你錯過了3件事.Main.javaimport org.eclipse.jetty.server.Server;import org.eclipse.jetty.server.handler.ContextHandler;import org.eclipse.jetty.server.session.HashSessionIdManager;import org.eclipse.jetty.server.session.HashSessionManager;import org.eclipse.…

什么是鏈表

鏈表是一種物理存儲單元上非連續、非順序的存儲結構&#xff0c;數據元素的邏輯順序是通過鏈表中的指針鏈接次序實現的。鏈表由一系列結點&#xff08;鏈表中每一個元素稱為結點&#xff09;組成&#xff0c;結點可以在運行時動態生成。每個結點包括兩個部分&#xff1a;一個是…

C# 基礎:Sealed、new、virtual、abstract、override的理解

目錄 1、sealed 2、new 3、virtual 4、abstract 5、override 1、sealed 密封類不能被繼承&#xff0c;密封方法可以重寫基類中的方法&#xff0c;但其本身不能在任何派生類&#xff08;子類&#xff09;中 進一步重寫。當應用于屬性或者方法時&#xff0c;sealed 修飾符必須始終…

梁興珍 java_數據結構與算法_Java語言

第1章 綜述1.1 數據結構和算法能起到什么作用&#xff1f;1.2 數據結構的概述1.3 算法的概述1.4 一些定義1.5 面向對象編程1.6 軟件工程1.7 對于C程序員的Java1.8 Java數據結構的類庫第2章 數組2.1 Array專題Applet2.2 Java中數組的基礎知識2.3 將程序劃分成類2.4 類接口2.5 Or…

Yii 2.0: yii2-highcharts-widget創建餅狀圖

安裝 The preferred way to install this extension is through composer. 項目根目錄下執行&#xff1a; php composer.phar require --prefer-dist miloschuman/yii2-highcharts-widget "*"或者在composer.json中添加 "miloschuman/yii2-highcharts-widget&qu…