Java語言中,把異常分為兩類:
受檢異常: 這類異常必須在throws子句中被顯式拋出或者在方法內被捕獲。例如,IOException異常或ClassNotFoundException異常。
非受檢異常: 這類異常不需要顯式拋出或捕獲。例如,NumberFormatException異常。
當一個受檢異常在Thread對象的run()方法中被拋出時,我們必須捕獲并處理它,因為run()方法不能拋出異常。而一個非受檢異常在Thread對象的run()方法中被拋出時,默認的行為是在控制臺打印出堆棧跟蹤信息然后退出程序。
幸運的是,Java為我們提供了一種機制,專門用于處理由Thread對象拋出的非受檢異常,以避免程序的退出。
在本節,我們用示例來演示這種機制。
知其然
按照下面所示步驟來實現我們的示例。
1.首先,我們需要實現一個用于處理非受檢異常的類。這個類必須實現UncaughtExceptionHandler類,實現在該接口中聲明的uncaughtException()方法。在本例中,該類名為ExceptionHandler,uncaughtException()方法將異常以及拋出異常的線程信息打印出來。代碼如下:
public class ExceptionHandler implements Thread.UncaughtExceptionHandler {
@Override
public void uncaughtException(Thread t, Throwable e) {
System.out.printf("An exception has been captured\n");
System.out.printf("Thread: %s ", t.getId());
System.out.printf("Exception: %s: %s ", e.getClass().getName(),
e.getMessage());
System.out.printf("Stack Trace: ");
e.printStackTrace(System.out);
System.out.printf("Thread status: %s ", t.getState());
}
}
2.實現一個可以拋出非受檢異常的類,稱為Task,實現Runnable接口,實現run()方法,特意編碼一段可以產生非受檢異常的代碼,例如,將字符串轉換成數字。代碼如下:
public class Task implements Runnable {
@Override
public void run() {
int numero = Integer.parseInt("diguage.com");
}
}
3.創建程序的主類,Main類,然后實現main()方法。代碼如下:
public class Main {
public static void main(String[] args) {
4.創建Task對象,并且創建一個Thread對象來執行之。使用setUncaughtExceptionHandler() 方法設置非受檢異常的處理類。然后,啟動線程。代碼如下:
Task task = new Task();
Thread thread = new Thread(task);
thread.setUncaughtExceptionHandler(new ExceptionHandler());
thread.start();
5.運行示例,查看結果。
知其所以然
從下面的輸出片段可以看出異常執行的結果。異常被拋出,然后被處理類捕獲并將異常信息打印到了控制臺。
An exception has been captured
Thread: 9
Exception: java.lang.NumberFormatException: For input string: "diguage.com"
Stack Trace:
java.lang.NumberFormatException: For input string: "diguage.com"
at java.lang.NumberFormatException.forInputString(NumberFormatException.java:65)
at java.lang.Integer.parseInt(Integer.java:492)
at java.lang.Integer.parseInt(Integer.java:527)
at com.diguage.books.concurrencycookbook.chapter1.recipe8.Task.run(Task.java:13)
at java.lang.Thread.run(Thread.java:722)
Thread status: RUNNABLE
Process finished with exit code 0
當一個線程拋出一個異常,并且該異常(這里特指非受檢異常)沒有捕獲時,Java虛擬機會檢查是否通過相應方法設置非受檢異常處理類,如果以已經設置過,則調用uncaughtException()方法,并將線程和異常作為參數傳遞給方法。
如果沒有設置處理類,Java虛擬機就會在控制臺將堆棧跟蹤信息打印出來,然后退出程序。
永無止境
Thread類還有一個和非受檢異常處理相關的方法。這就是靜態方法setDefaultUncaughtExceptionHandler(),該方法可以設置程序中所有線程的非受檢異常的處理類。
當線程中拋出一個未捕獲的異常時,Java虛擬機會從三個地方尋找異常處理類:
首先,從線程對象中查找異常處理類,這就是我們本節所學內容。如不存在,則從線程所在的線程組(ThreadGroup)中查找異常處理類。關于這部分內容,以后會專門講解。如果還是不存在,則查找上面剛剛提到的程序默認異常處理類。
如果上面提到的異常處理都不存在,則Java虛擬機將異常的堆棧跟蹤信息打印到控制臺,然后退出程序。
拿來主義
本文是從 《Java 7 Concurrency Cookbook》 (D瓜哥竊譯為 《Java7并發示例集》 )翻譯而來,僅作為學習資料使用。沒有授權,不得用于任何商業行為。
小有所成
ExceptionHandler類的完整代碼
package com.diguage.books.concurrencycookbook.chapter1.recipe8;
/**
* 非受檢異常處理類
* Date: 2013-09-22
* Time: 23:11
*/
public class ExceptionHandler implements Thread.UncaughtExceptionHandler {
@Override
public void uncaughtException(Thread t, Throwable e) {
System.out.printf("An exception has been captured ");
System.out.printf("Thread: %s ", t.getId());
System.out.printf("Exception: %s: %s ", e.getClass().getName(),
e.getMessage());
System.out.printf("Stack Trace: ");
e.printStackTrace(System.out);
System.out.printf("Thread status: %s ", t.getState());
}
}
Task類的完整代碼
package com.diguage.books.concurrencycookbook.chapter1.recipe8;
/**
* 異常生成類
* Date: 2013-09-22
* Time: 23:18
*/
public class Task implements Runnable {
@Override
public void run() {
int numero = Integer.parseInt("diguage.com");
}
}
Main類的完整代碼
package com.diguage.books.concurrencycookbook.chapter1.recipe8;
/**
* 示例的主類
* Date: 2013-09-22
* Time: 23:20
*/
public class Main {
public static void main(String[] args) {
Task task = new Task();
Thread thread = new Thread(task);
thread.setUncaughtExceptionHandler(new ExceptionHandler());
thread.start();
}
}