2010-02-26 28 views
19

我目前正在使用Tomcat,Spring和JAVA構建一個應用程序。我正在使用Log4J作爲我的日誌記錄庫。我目前正在將所有內容記錄到文本文件中。我遇到的其中一個問題是RuntimeExceptions未被記錄到任何文件。我想知道是否有辦法記錄可能拋到我的應用程序日誌文件中的所有RuntimeExceptions。如果不是,它有可能讓它登錄到另一個日誌文件?有沒有一個標準的方法來做到這一點?如果是的話,在Tomcat中運行應用程序時有沒有一種標準的方法來做到這一點?日誌運行時Java中使用log4j的異常

非常感謝您的幫助!

+2

你的意思是「未捕獲」 RuntimeExceptions? – 2010-02-26 20:43:43

+0

是的我的意思是「未被捕獲」RuntimeExceptions – DkS 2010-02-26 20:58:26

回答

21

我不確定這是你在找什麼,但有一個終止線程的異常處理程序。它是任何異常的處理程序,不會被線程的目標顯式捕獲。

默認的「未捕獲的異常處理程序」只需調用就可以將堆棧跟蹤打印到System.err。但是,你可以replace這個跟自己UncaughtExceptionHandler它記錄的異常,而不是結果到log4j:

class Log4jBackstop implements Thread.UncaughtExceptionHandler { 

    private static Logger log = Logger.getLogger(Log4jBackstop.class); 

    public void uncaughtException(Thread t, Throwable ex) { 
    log.error("Uncaught exception in thread: " + t.getName(), ex); 
    } 

} 

如果您使用的是執行框架,其中傳遞一個Runnable對象,它的線程可能有自己的catch塊防止到達未捕獲的異常處理程序的異常。如果你想趕上Runtime例外存在,做一個方式,它是包裝每一個任務在日誌的包裝,像這樣:

class Log4jWrapper { 

    private final Logger log; 

    private final Runnable target; 

    Log4jWrapper(Logger log, Runnable target) { 
    this.log = Objects.requireNonNull(log); 
    this.target = Objects.requireNonNull(target); 
    } 

    public void run() { 
    try { 
     target.run(); 
    } catch(RuntimeException ex) { 
     log.error("Uncaught exception.", ex); 
     throw ex; 
    } 
    } 

} 

... 

Runnable realTask = ...; 
executor.submit(new Log4jWrapper(Logger.getLogger(Whatever.class), realTask)); 
+0

通常這是如何處理這個RuntimeExceptions在應用程序?我想我想知道這是否是處理這個問題的標準方法。 – DkS 2010-02-26 22:42:50

+0

另外我使用Spring API的ThreadPoolTask​​Executor,這仍然可以使用這個類來完成嗎? – DkS 2010-02-26 22:54:38

+0

如果在Runnable上有RuntimeException,這很好用,但是如果ThreadPoolTask​​Executor有一個Exception呢?說它拋出一個RunTimeException,因爲池配置正確....這將仍然打印到控制檯,不會被記錄....任何建議? – DkS 2010-02-27 23:13:53

12

一個記錄未捕獲RuntimeExceptions(或與此有關的任何未捕獲將Throwable)的方法是創建一個實現Thread.UncaughtExceptionHandler類。這個類的uncaughtException()方法只會將異常詳細信息寫入日誌。您可以在JVM調用時未捕獲的RuntimeException通過增加線

Thread.setDefaultUncaughtExceptionHandler(new MyExceptionHandler()); 

你的代碼終止線程這個異常處理程序。