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

2022-09-01 20:30:34

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

提前感谢您的帮助!


答案 1

我不确定这是否是您要查找的,但是有一个用于终止线程的异常的处理程序。它是线程目标未显式捕获的任何异常的处理程序。

默认的“未捕获异常处理程序”只是调用 将堆栈跟踪打印到 。但是,您可以将其替换为您自己的UncaughtExceptionHandler,该Handler将异常记录到log4j中:printStackTrace()ThrowableSystem.err

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);
  }

}

如果使用的是将对象传递到的执行器框架,则其线程可能具有自己的块,以防止异常到达未捕获的异常处理程序。如果你想在那里捕获异常,一种方法是将每个任务包装在日志记录包装器中,如下所示:RunnablecatchRuntime

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));

答案 2

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

Thread.setDefaultUncaughtExceptionHandler(new MyExceptionHandler());

添加到您的代码中。


推荐