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/n", t.getId());
System.out.printf("Exception: %s: %s/n", e.getClass().getName(),
e.getMessage());
System.out.printf("Stack Trace: /n");
e.printStackTrace(System.out);
System.out.printf("Thread status: %s/n", 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/n");
System.out.printf("Thread: %s/n", t.getId());
System.out.printf("Exception: %s: %s/n", e.getClass().getName(),
e.getMessage());
System.out.printf("Stack Trace: /n");
e.printStackTrace(System.out);
System.out.printf("Thread status: %s/n", 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();
}
}