Thread的run方法是不拋出任何檢查型異常(checked exception)的,但是它自身卻可能因為一個異常而被終止,導致這個線程的終結。最麻煩的是,在線程中拋出的異常即使使用try...catch也無法截獲,因此可能導致一些問題出現,比如異常的時候無法回收一些系統資源,或者沒有關閉當前的連接等等。
JDK5.0之前,不能為單獨的Thread設置UncaughtExceptionHandler,也不能指定一個默認的UncaughtExceptionHandler。為了可以設置一個UncaughtExceptionHandler,需要去繼承ThreadGroup並覆寫uncaughtException方法。
在JDK5.0中,我們通過Thread的實例方法setUncaughtExceptionHandler,可以為任何一個Thread設置一個UncaughtExceptionHandler。當然你也可以為所有Thread設置一個默認的UncaughtExceptionHandler,通過調用Thread.setDefaultUncaughtExceptionHandler(Thread.UncaughtExceptionHandler eh)方法,這是Thread的一個static方法。
定義一個Handler類必須實現Thread.UncaughtExceptionHandler接口的void uncaughtException(Thread t, Throwable e)方法。如果不設置一個Handler,那么單個Thread的Handler是null。但是,如果這個單個線程是ThreadGroup中的一個Thread,那么這個線程將使用ThreadGroup的UncaughtExceptionHandler。ThreadGroup自身已經實現了Thread.UncaughtExceptionHandler接口。
這樣就夠了
uncaughtException(Thread a, Throwable e)可以拿到Thread,所以在uncaughtException釋放相關資源是最好的辦法。
總之,JDK5.0中Thread及其相關的輔助功能得到了加強,為我們提供了很多便利和安全的解決方案:)
import java.lang.Thread.UncaughtExceptionHandler; public class ThreadTest { public static void main(String[] args) { ErrHandler handle = null; ThreadA a = null; a = new ThreadA(); handle = new ErrHandler(); a.setUncaughtExceptionHandler(handle);// 加入定義的ErrHandler a.start(); } } /** * 自定義的一個UncaughtExceptionHandler */ class ErrHandler implements UncaughtExceptionHandler { /** * 這里可以做任何針對異常的處理,比如記錄日志等等 */ public void uncaughtException(Thread a, Throwable e) { System.out.println("This is:" + a.getName() + ",Message:" + e.getMessage()); e.printStackTrace(); } } /** * 擁有UncaughtExceptionHandler的線程 */ class ThreadA extends Thread { public ThreadA() { } public void run() { double i = 12 / 0;// 拋出異常的地方 } }