使用log4j在Java中记录运行时异常

时间:2010-02-26 20:35:48

标签: java spring tomcat logging log4j

我目前正在使用Tomcat,Spring和JAVA构建应用程序。我使用Log4J作为我的日志库。我目前正在将所有内容记录到文本文件中。我遇到的一个问题是RuntimeExceptions没有被记录到任何文件中。我想知道是否有办法记录可能抛出到我的应用程序日志文件的所有RuntimeExceptions。如果不是可以让它登录到另一个日志文件?有没有标准的方法来做到这一点?如果是这样,在Tomcat中运行应用程序时有一种标准的方法吗?

提前感谢您的帮助!

3 个答案:

答案 0 :(得分:21)

我不确定这是否是您正在寻找的,但是有一个终止线程的异常处理程序。它是线程目标未明确捕获的任何异常的处理程序。

默认的“未捕获的异常处理程序”只需调用printStackTrace()上的Throwable即可将堆栈跟踪打印到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));

答案 1 :(得分:13)

记录未捕获的RuntimeExceptions(或任何未捕获的Throwables)的一种方法是创建一个实现Thread.UncaughtExceptionHandler的类。此类的uncaughtException()方法只是将异常详细信息写入日志。每当未捕获的RuntimeException通过添加行终止线程时,您可以使JVM调用此异常处理程序

Thread.setDefaultUncaughtExceptionHandler(new MyExceptionHandler());

代码。

答案 2 :(得分:2)

您可以尝试重定向StdErr以使用记录器。这应该输出任何写入std err的结果(应包括未处理的异常)

这篇博客文章简要介绍了如何重定向到您的记录器之外的FileHandler。

https://blogs.oracle.com/nickstephen/entry/java_redirecting_system_out_and