Log runtime Exceptions in Java using log4j

前端 未结 3 1571
终归单人心
终归单人心 2020-12-03 04:56

I am currently building an application using Tomcat, Spring and JAVA. I am using Log4J as my logging library. I currently am logging everything to a text file

相关标签:
3条回答
  • 2020-12-03 05:32

    You can try redirecting the StdErr to use your logger. This should output the result of anything being written to your std err (which should include unhandled exceptions)

    This blog post gives a good rundown on how to do the redirection into a FileHandler that is apart of your logger:

    • Redirecting System.out and System.err to a rolling log file (Nick Stephen's blog)
    0 讨论(0)
  • 2020-12-03 05:35

    One way to log uncaught RuntimeExceptions (or any uncaught Throwables for that matter) is to create a class that implements Thread.UncaughtExceptionHandler. This class's uncaughtException() method would simply write the exception details to the log. You can make the JVM call this exception handler whenever an uncaught RuntimeException terminates a thread by adding the line

    Thread.setDefaultUncaughtExceptionHandler(new MyExceptionHandler());
    

    to your code.

    0 讨论(0)
  • 2020-12-03 05:45

    I'm not sure if this is what you are looking for, but there is a handler for exceptions that terminate threads. It is a handler for any exception that is not caught explicitly by the target of the thread.

    The default "uncaught exception handler" simply calls printStackTrace() on the Throwable to print the stack trace to System.err. However, you could replace this with your own UncaughtExceptionHandler that logs the exception to log4j instead:

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

    If you are using an executor framework to which you pass a Runnable object, its threads probably have their own catch block that prevent exceptions from reaching the uncaught exception handler. If you want to catch Runtime exceptions there, one way to do it is to wrap each task in a logging wrapper, like this:

    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 讨论(0)
提交回复
热议问题