[深入理解java異常處理機制](http://blog.csdn.net/hguisu/article/details/6155636/)
[TOC=1,3]
##
## 1. 引子
? ? ? ?try…catch…finally恐怕是大家再熟悉不過的語句了,而且感覺用起來也是很簡單,邏輯上似乎也是很容易理解。不過,我親自體驗的“教訓”告訴我,這個東西可不是想象中的那么簡單、聽話。不信?那你看看下面的代碼,“猜猜”它執行后的結果會是什么?不要往后看答案、也不許執行代碼看真正答案哦。如果你的答案是正確,那么這篇文章你就不用浪費時間看啦。
1. package?Test;??
2. ??
3. public?class?TestException?{??
4. ????public?TestException()?{??
5. ????}??
6. ??
7. ????boolean?testEx()?throws?Exception?{??
8. ????????boolean?ret?=?true;??
9. ????????try?{??
10. ????????????ret?=?testEx1();??
11. ????????}?catch?(Exception?e)?{??
12. ????????????System.out.println("testEx,?catch?exception");??
13. ????????????ret?=?false;??
14. ????????????throw?e;??
15. ????????}?finally?{??
16. ????????????System.out.println("testEx,?finally;?return?value="?+?ret);??
17. ????????????return?ret;??
18. ????????}??
19. ????}??
20. ??
21. ????boolean?testEx1()?throws?Exception?{??
22. ????????boolean?ret?=?true;??
23. ????????try?{??
24. ????????????ret?=?testEx2();??
25. ????????????if?(!ret)?{??
26. ????????????????return?false;??
27. ????????????}??
28. ????????????System.out.println("testEx1,?at?the?end?of?try");??
29. ????????????return?ret;??
30. ????????}?catch?(Exception?e)?{??
31. ????????????System.out.println("testEx1,?catch?exception");??
32. ????????????ret?=?false;??
33. ????????????throw?e;??
34. ????????}?finally?{??
35. ????????????System.out.println("testEx1,?finally;?return?value="?+?ret);??
36. ????????????return?ret;??
37. ????????}??
38. ????}??
39. ??
40. ????boolean?testEx2()?throws?Exception?{??
41. ????????boolean?ret?=?true;??
42. ????????try?{??
43. ????????????int?b?=?12;??
44. ????????????int?c;??
45. ????????????for?(int?i?=?2;?i?>=?-2;?i--)?{??
46. ????????????????c?=?b?/?i;??
47. ????????????????System.out.println("i="?+?i);??
48. ????????????}??
49. ????????????return?true;??
50. ????????}?catch?(Exception?e)?{??
51. ????????????System.out.println("testEx2,?catch?exception");??
52. ????????????ret?=?false;??
53. ????????????throw?e;??
54. ????????}?finally?{??
55. ????????????System.out.println("testEx2,?finally;?return?value="?+?ret);??
56. ????????????return?ret;??
57. ????????}??
58. ????}??
59. ??
60. ????public?static?void?main(String[]?args)?{??
61. ????????TestException?testException1?=?new?TestException();??
62. ????????try?{??
63. ????????????testException1.testEx();??
64. ????????}?catch?(Exception?e)?{??
65. ????????????e.printStackTrace();??
66. ????????}??
67. ????}??
68. }??
你的答案是什么?是下面的答案嗎?
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
如果你的答案真的如上面所說,那么你錯啦。^_^,那就建議你仔細看一看這篇文章或者拿上面的代碼按各種不同的情況修改、執行、[測試](http://lib.csdn.net/base/softwaretest "軟件測試知識庫"),你會發現有很多事情不是原來想象中的那么簡單的。現在公布正確答案:
i=2
i=1
testEx2, catch exception
testEx2, finally; return value=false
testEx1, finally; return value=false
testEx, finally; return value=false
注意說明:
finally語句塊不應該出現 應該出現return。上面的return ret最好是其他語句來處理相關邏輯。
##
?## 2.[Java](http://lib.csdn.net/base/javase "Java SE知識庫")異常
異常指不期而至的各種狀況,如:文件找不到、網絡連接失敗、非法參數等。異常是一個事件,它發生在程序運行期間,干擾了正常的指令流程。Java通 過API中Throwable類的眾多子類描述各種不同的異常。因而,Java異常都是對象,是Throwable子類的實例,描述了出現在一段編碼中的 錯誤條件。當條件生成時,錯誤將引發異常。
? ? ??Java異常類層次結構圖:
? ? ? ??
? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ??圖1?Java異常類層次結構圖
? ? ? ? 在 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異常,一般情況下不自定義檢查異常。
**? ??**
##
?## 4.處理異常機制
? ? ? ??在 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。
### 4.1 捕獲異常:try、catch 和 finally
**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. public?class?TestException?{??
2. ????public?static?void?main(String[]?args)?{??
3. ????????int?a?=?6;??
4. ????????int?b?=?0;??
5. ????????try?{?//?try監控區域??
6. ??????????????
7. ????????????if?(b?==?0)?throw?new?ArithmeticException();?//?通過throw語句拋出異常??
8. ????????????System.out.println("a/b的值是:"?+?a?/?b);??
9. ????????}??
10. ????????catch?(ArithmeticException?e)?{?//?catch捕捉異常??
11. ????????????System.out.println("程序出現異常,變量b不能為0。");??
12. ????????}??
13. ????????System.out.println("程序正常結束。");??
14. ????}??
15. }??
運行結果:程序出現異常,變量b不能為0。
? ? ? ? ? ? ? ? ? ? 程序正常結束。
? ? ? ? 例1? 在try監控區域通過if語句進行判斷,當“除數為0”的錯誤條件成立時引發ArithmeticException異常,創建 ArithmeticException異常對象,并由throw語句將異常拋給Java運行時系統,由系統尋找匹配的異常處理器catch并運行相應異 常處理代碼,打印輸出“程序出現異常,變量b不能為0。”try-catch語句結束,繼續程序流程。
? ? ? ? 事實上,“除數為0”等ArithmeticException,是RuntimException的子類。而運行時異常將由運行時系統自動拋出,不需要使用throw語句。
例2??捕捉運行時系統自動拋出“除數為0”引發的ArithmeticException異常。
1. ????public?static?void?main(String[]?args)?{??
2. ????????int?a?=?6;??
3. ????????int?b?=?0;??
4. ????????try?{??
5. ????????????System.out.println("a/b的值是:"?+?a?/?b);??
6. ????????}?catch?(ArithmeticException?e)?{??
7. ????????????System.out.println("程序出現異常,變量b不能為0。");??
8. ????????}??
9. ????????System.out.println("程序正常結束。");??
10. ????}??
11. }??
運行結果:程序出現異常,變量b不能為0。
? ? ? ? ? ? ? ? ? 程序正常結束。
例2? 中的語句:
System.out.println("a/b的值是:" + a/b);
? ? ? 在運行中出現“除數為0”錯誤,引發ArithmeticException異常。運行時系統創建異常對象并拋出監控區域,轉而匹配合適的異常處理器catch,并執行相應的異常處理代碼。
? ? ? 由于檢查運行時異常的代價遠大于捕捉異常所帶來的益處,運行時異常不可查。Java編譯器允許忽略運行時異常,一個方法可以既不捕捉,也不聲明拋出運行時異常。
例3??不捕捉、也不聲明拋出運行時異常。
1. public?class?TestException?{??
2. ????public?static?void?main(String[]?args)?{??
3. ????????int?a,?b;??
4. ????????a?=?6;??
5. ????????b?=?0;?//?除數b?的值為0??
6. ????????System.out.println(a?/?b);??
7. ????}??
8. }??
運行結果:
Exception in thread "main" java.lang.ArithmeticException: / by zero
at Test.TestException.main(TestException.java:8)
例4? 程序可能存在除數為0異常和數組下標越界異常。
1. public?class?TestException?{??
2. ????public?static?void?main(String[]?args)?{??
3. ????????int[]?intArray?=?new?int[3];??
4. ????????try?{??
5. ????????????for?(int?i?=?0;?i?
6. ????????????????intArray[i]?=?i;??
7. ????????????????System.out.println("intArray["?+?i?+?"]?=?"?+?intArray[i]);??
8. ????????????????System.out.println("intArray["?+?i?+?"]模?"?+?(i?-?2)?+?"的值:??"??
9. ????????????????????????+?intArray[i]?%?(i?-?2));??
10. ????????????}??
11. ????????}?catch?(ArrayIndexOutOfBoundsException?e)?{??
12. ????????????System.out.println("intArray數組下標越界異常。");??
13. ????????}?catch?(ArithmeticException?e)?{??
14. ????????????System.out.println("除數為0異常。");??
15. ????????}??
16. ????????System.out.println("程序正常結束。");??
17. ????}??
18. }??
運行結果:
intArray[0] = 0
intArray[0]模 -2的值:? 0
intArray[1] = 1
intArray[1]模 -1的值:? 0
intArray[2] = 2
除數為0異常。
程序正常結束。
? ? ??例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子句應該放在 最后面,否則可能會屏蔽其后的特定異常處理或引起編譯錯誤。
#### **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. ????//?無論是否發生異常,都將執行的語句塊??
9. }??
例5??帶finally子句的異常處理程序。
1. public?class?TestException?{??
2. ????public?static?void?main(String?args[])?{??
3. ????????int?i?=?0;??
4. ????????String?greetings[]?=?{?"?Hello?world?!",?"?Hello?World?!!?",??
5. ????????????????"?HELLO?WORLD?!!!"?};??
6. ????????while?(i?4)?{??
7. ????????????try?{??
8. ????????????????//?特別注意循環控制變量i的設計,避免造成無限循環??
9. ????????????????System.out.println(greetings[i++]);??
10. ????????????}?catch?(ArrayIndexOutOfBoundsException?e)?{??
11. ????????????????System.out.println("數組下標越界異常");??
12. ????????????}?finally?{??
13. ????????????????System.out.println("--------------------------");??
14. ????????????}??
15. ????????}??
16. ????}??
17. }??
運行結果:
Hello world !
--------------------------
Hello World !!
--------------------------
HELLO WORLD !!!
--------------------------
數組下標越界異常
--------------------------
? ? ?在例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\. try-catch-finally 規則(**[**異常處理語句的語法規則**](http://book.51cto.com/art/201009/227791.htm "16.2.5 異常處理語句的語法規則")**):**
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 塊中拋出一個未處理的異常;計算機斷電、失火、或遭遇病毒攻擊。
#### **4\. try、catch、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語句塊的執行:

? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? 圖2 ?圖示try、catch、finally語句塊的執行
### **4.2 拋出異常**
? ? ? 任何Java代碼都可以拋出異常,如:自己編寫的代碼、來自Java開發環境包中代碼,或者Java運行時系統。無論是誰,都可以通過Java的throw語句拋出異常。從方法中拋出的任何異常都必須使用throws子句。
#### 1\. throws拋出異常
? ?如果一個方法可能會出現異常,但沒有能力處理這種異常,可以在方法聲明處用throws子句來聲明拋出異常。例如汽車在運行時可能會出現故障,汽車本身沒辦法處理這個故障,那就讓開車的人來處理。
? ? ?throws語句用在方法定義時聲明該方法要拋出的異常類型,如果拋出的是Exception異常類型,則該方法被聲明為拋出所有的異常。多個異常可使用逗號分割。throws語句的語法格式為:
1. methodname?throws?Exception1,Exception2,..,ExceptionN??
2. {??
3. }??
????方法名后的throws Exception1,Exception2,...,ExceptionN 為聲明要拋出的異常列表。當方法拋出異常列表的異常時,方法將不對這些類型及其子類類型的異常作處理,而拋向調用該方法的方法,由他去處理。例如:
1. import?java.lang.Exception;??
2. public?class?TestException?{??
3. ????static?void?pop()?throws?NegativeArraySizeException?{??
4. ????????//?定義方法并拋出NegativeArraySizeException異常??
5. ????????int[]?arr?=?new?int[-3];?//?創建數組??
6. ????}??
7. ??
8. ????public?static?void?main(String[]?args)?{?//?主方法??
9. ????????try?{?//?try語句處理異常信息??
10. ????????????pop();?//?調用pop()方法??
11. ????????}?catch?(NegativeArraySizeException?e)?{??
12. ????????????System.out.println("pop()方法拋出的異常");//?輸出異常信息??
13. ????????}??
14. ????}??
15. ??
16. }??
? ? 使用throws關鍵字將異常拋給調用者后,如果調用者不想處理該異常,可以繼續向上拋出,但最終要有能夠處理該異常的調用者。
? ? pop方法沒有處理異常NegativeArraySizeException,而是由main函數來處理。
? ??**Throws拋出異常的規則:**
? ? 1) 如果是不可查異常(unchecked exception),即Error、RuntimeException或它們的子類,那么可以不使用throws關鍵字來聲明要拋出的異常,編譯仍能順利通過,但在運行時會被系統拋出。
? ? 2)必須聲明方法可拋出的任何可查異常(checked exception)。即如果一個方法可能出現受可查異常,要么用try-catch語句捕獲,要么用throws子句聲明將它拋出,否則會導致編譯錯誤
? ? 3)僅當拋出了異常,該方法的調用者才必須處理或者重新拋出該異常。當方法的調用者無力處理該異常的時候,應該繼續拋出,而不是囫圇吞棗。
? ? 4)調用方法必須遵循任何可查異常的處理和聲明規則。若覆蓋一個方法,則不能聲明與覆蓋方法不同的異常。聲明的任何異常必須是被覆蓋方法所聲明異常的同類或子類。
? ? 例如:
1. void?method1()?throws?IOException{}??//合法????
2. ???
3. //編譯錯誤,必須捕獲或聲明拋出IOException????
4. void?method2(){????
5. ??method1();????
6. }????
7. ???
8. //合法,聲明拋出IOException????
9. void?method3()throws?IOException?{????
10. ??method1();????
11. }????
12. ???
13. //合法,聲明拋出Exception,IOException是Exception的子類????
14. void?method4()throws?Exception?{????
15. ??method1();????
16. }????
17. ???
18. //合法,捕獲IOException????
19. void?method5(){????
20. ?try{????
21. ????method1();????
22. ?}catch(IOException?e){…}????
23. }????
24. ???
25. //編譯錯誤,必須捕獲或聲明拋出Exception????
26. void?method6(){????
27. ??try{????
28. ????method1();????
29. ??}catch(IOException?e){throw?new?Exception();}????
30. }????
31. ???
32. //合法,聲明拋出Exception????
33. void?method7()throws?Exception{????
34. ?try{????
35. ??method1();????
36. ?}catch(IOException?e){throw?new?Exception();}????
37. }???
? ??**?判斷一個方法可能會出現異常的依據如下:**
? ? ?1)方法中有throw語句。例如,以上method7()方法的catch代碼塊有throw語句。
? ? ?2)調用了其他方法,其他方法用throws子句聲明拋出某種異常。例如,method3()方法調用了method1()方法,method1()方法聲明拋出IOException,因此,在method3()方法中可能會出現IOException。
#### 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. package?Test;??
2. import?java.lang.Exception;??
3. public?class?TestException?{??
4. ????static?int?quotient(int?x,?int?y)?throws?MyException?{?//?定義方法拋出異常??
5. ????????if?(y?0)?{?//?判斷參數是否小于0??
6. ????????????throw?new?MyException("除數不能是負數");?//?異常信息??
7. ????????}??
8. ????????return?x/y;?//?返回值??
9. ????}??
10. ????public?static?void?main(String?args[])?{?//?主方法??
11. ????????int??a?=3;??
12. ????????int??b?=0;???
13. ????????try?{?//?try語句包含可能發生異常的語句??
14. ????????????int?result?=?quotient(a,?b);?//?調用方法quotient()??
15. ????????}?catch?(MyException?e)?{?//?處理自定義異常??
16. ????????????System.out.println(e.getMessage());?//?輸出異常信息??
17. ????????}?catch?(ArithmeticException?e)?{?//?處理ArithmeticException異常??
18. ????????????System.out.println("除數不能為0");?//?輸出提示信息??
19. ????????}?catch?(Exception?e)?{?//?處理其他異常??
20. ????????????System.out.println("程序發生了其他的異常");?//?輸出提示信息??
21. ????????}??
22. ????}??
23. ??
24. }??
25. class?MyException?extends?Exception?{?//?創建自定義異常類??
26. ????String?message;?//?定義String類型變量??
27. ????public?MyException(String?ErrorMessagr)?{?//?父類方法??
28. ????????message?=?ErrorMessagr;??
29. ????}??
30. ??
31. ????public?String?getMessage()?{?//?覆蓋getMessage()方法??
32. ????????return?message;??
33. ????}??
34. }??
**4.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語句的順序不可掉換。
### 4.4 Throwable類中的常用方法
注意:catch關鍵字后面括號中的Exception類型的參數e。Exception就是try代碼塊傳遞給catch代碼塊的變量類型,e就是變量名。catch代碼塊中語句"e.getMessage();"用于輸出錯誤性質。通常異常處理常用3個函數來獲取異常的有關信息:
? ? ?getCause():返回拋出異常的原因。如果 cause 不存在或未知,則返回 null。
getMeage():返回異常的消息信息。
printStackTrace():對象的堆棧跟蹤輸出至錯誤輸出流,作為字段 System.err 的值。
? ? ?有時為了簡單會忽略掉catch語句后的代碼,這樣try-catch語句就成了一種擺設,一旦程序在運行過程中出現了異常,就會忽略處理異常,而錯誤發生的原因很難查找。
##
?## 5.Java常見異常
在Java中提供了一些異常用來描述經常發生的錯誤,對于這些異常,有的需要程序員進行捕獲處理或聲明拋出,有的是由Java虛擬機自動進行捕獲處理。Java中常見的異常類:
**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?非法參數異常
****2.IOException**
> IOException:操作輸入流和輸出流時可能出現的異常。
>
> EOFException ? 文件已結束異常
>
> FileNotFoundException ? 文件未找到異常
****3.其他**
> ClassCastException ? ?類型轉換異常類
>
> ArrayStoreException ?數組中包含不兼容的值拋出的異常
>
> SQLException ? 操作[數據庫](http://lib.csdn.net/base/mysql "MySQL知識庫")異常類
>
> NoSuchFieldException ? 字段未找到異常
>
> NoSuchMethodException ? 方法未找到拋出的異常
>
> NumberFormatException ? ?字符串轉換為數字拋出的異常
>
> StringIndexOutOfBoundsException 字符串索引超出范圍拋出的異常
>
> IllegalAccessException ?不允許訪問某類異常
>
> InstantiationException ?當應用程序試圖使用Class類中的newInstance()方法創建一個類的實例,而指定的類對象無法被實例化時,拋出該異常
**? ??**
##
## ?6.自定義異常
使用Java內置的異常類可以描述在編程時出現的大部分異常情況。除此之外,用戶還可以自定義異常。用戶自定義異常類,只需繼承Exception類即可。
? ? 在程序中使用自定義異常類,大體可分為以下幾個步驟。
(1)創建自定義異常類。
(2)在方法中通過throw關鍵字拋出異常對象。
(3)如果在當前拋出異常的方法中處理異常,可以使用try-catch語句捕獲并處理;否則在方法的聲明處通過throws關鍵字指明要拋出給方法調用者的異常,繼續進行下一步操作。
(4)在出現異常方法的調用者中捕獲并處理異常。
在上面的“使用throw拋出異常”例子已經提到了。
- JVM
- 深入理解Java內存模型
- 深入理解Java內存模型(一)——基礎
- 深入理解Java內存模型(二)——重排序
- 深入理解Java內存模型(三)——順序一致性
- 深入理解Java內存模型(四)——volatile
- 深入理解Java內存模型(五)——鎖
- 深入理解Java內存模型(六)——final
- 深入理解Java內存模型(七)——總結
- Java內存模型
- Java內存模型2
- 堆內內存還是堆外內存?
- JVM內存配置詳解
- Java內存分配全面淺析
- 深入Java核心 Java內存分配原理精講
- jvm常量池
- JVM調優總結
- JVM調優總結(一)-- 一些概念
- JVM調優總結(二)-一些概念
- VM調優總結(三)-基本垃圾回收算法
- JVM調優總結(四)-垃圾回收面臨的問題
- JVM調優總結(五)-分代垃圾回收詳述1
- JVM調優總結(六)-分代垃圾回收詳述2
- JVM調優總結(七)-典型配置舉例1
- JVM調優總結(八)-典型配置舉例2
- JVM調優總結(九)-新一代的垃圾回收算法
- JVM調優總結(十)-調優方法
- 基礎
- Java 征途:行者的地圖
- Java程序員應該知道的10個面向對象理論
- Java泛型總結
- 序列化與反序列化
- 通過反編譯深入理解Java String及intern
- android 加固防止反編譯-重新打包
- volatile
- 正確使用 Volatile 變量
- 異常
- 深入理解java異常處理機制
- Java異常處理的10個最佳實踐
- Java異常處理手冊和最佳實踐
- Java提高篇——對象克隆(復制)
- Java中如何克隆集合——ArrayList和HashSet深拷貝
- Java中hashCode的作用
- Java提高篇之hashCode
- 常見正則表達式
- 類
- 理解java類加載器以及ClassLoader類
- 深入探討 Java 類加載器
- 類加載器的工作原理
- java反射
- 集合
- HashMap的工作原理
- ConcurrentHashMap之實現細節
- java.util.concurrent 之ConcurrentHashMap 源碼分析
- HashMap的實現原理和底層數據結構
- 線程
- 關于Java并發編程的總結和思考
- 40個Java多線程問題總結
- Java中的多線程你只要看這一篇就夠了
- Java多線程干貨系列(1):Java多線程基礎
- Java非阻塞算法簡介
- Java并發的四種風味:Thread、Executor、ForkJoin和Actor
- Java中不同的并發實現的性能比較
- JAVA CAS原理深度分析
- 多個線程之間共享數據的方式
- Java并發編程
- Java并發編程(1):可重入內置鎖
- Java并發編程(2):線程中斷(含代碼)
- Java并發編程(3):線程掛起、恢復與終止的正確方法(含代碼)
- Java并發編程(4):守護線程與線程阻塞的四種情況
- Java并發編程(5):volatile變量修飾符—意料之外的問題(含代碼)
- Java并發編程(6):Runnable和Thread實現多線程的區別(含代碼)
- Java并發編程(7):使用synchronized獲取互斥鎖的幾點說明
- Java并發編程(8):多線程環境中安全使用集合API(含代碼)
- Java并發編程(9):死鎖(含代碼)
- Java并發編程(10):使用wait/notify/notifyAll實現線程間通信的幾點重要說明
- java并發編程-II
- Java多線程基礎:進程和線程之由來
- Java并發編程:如何創建線程?
- Java并發編程:Thread類的使用
- Java并發編程:synchronized
- Java并發編程:Lock
- Java并發編程:volatile關鍵字解析
- Java并發編程:深入剖析ThreadLocal
- Java并發編程:CountDownLatch、CyclicBarrier和Semaphore
- Java并發編程:線程間協作的兩種方式:wait、notify、notifyAll和Condition
- Synchronized與Lock
- JVM底層又是如何實現synchronized的
- Java synchronized詳解
- synchronized 與 Lock 的那點事
- 深入研究 Java Synchronize 和 Lock 的區別與用法
- JAVA編程中的鎖機制詳解
- Java中的鎖
- TreadLocal
- 深入JDK源碼之ThreadLocal類
- 聊一聊ThreadLocal
- ThreadLocal
- ThreadLocal的內存泄露
- 多線程設計模式
- Java多線程編程中Future模式的詳解
- 原子操作(CAS)
- [譯]Java中Wait、Sleep和Yield方法的區別
- 線程池
- 如何合理地估算線程池大小?
- JAVA線程池中隊列與池大小的關系
- Java四種線程池的使用
- 深入理解Java之線程池
- java并發編程III
- Java 8并發工具包漫游指南
- 聊聊并發
- 聊聊并發(一)——深入分析Volatile的實現原理
- 聊聊并發(二)——Java SE1.6中的Synchronized
- 文件
- 網絡
- index
- 內存文章索引
- 基礎文章索引
- 線程文章索引
- 網絡文章索引
- IOC
- 設計模式文章索引
- 面試
- Java常量池詳解之一道比較蛋疼的面試題
- 近5年133個Java面試問題列表
- Java工程師成神之路
- Java字符串問題Top10
- 設計模式
- Java:單例模式的七種寫法
- Java 利用枚舉實現單例模式
- 常用jar
- HttpClient和HtmlUnit的比較總結
- IO
- NIO
- NIO入門
- 注解
- Java Annotation認知(包括框架圖、詳細介紹、示例說明)