一尘不染

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

tomcat

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

预先感谢您的帮助!


阅读 373

收藏
2020-06-16

共1个答案

一尘不染

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

默认“未捕获的异常处理程序”只是调用printStackTrace()Throwable打印的堆栈跟踪System.err。但是,您可以将其替换为自己的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));
2020-06-16