2010-02-26 49 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()); 

你的代码终止线程这个异常处理程序。