重定向 System.out.println 至 log

2022-09-04 20:56:47

在我的项目测试套件中,有很大的用法

System.out.println 

我正在尝试将这些输出重定向到日志文件(通过配置或从单个点而不重构整个项目),以便在必要时可以禁用以提高性能。我正在使用log4j进行日志记录。有没有人知道这有可能?如果是这样,该怎么办?提前致谢。


答案 1

鉴于它最好替换 ,有时我们别无选择。无论如何,我已经为此做了一点实用程序:System.out.println()

SystemOutToSlf4j.enableForClass(MyClass.class)

然后,所有源自 MyClass 的 println 将被重定向到记录器。有关更多详细信息,请参阅此帖子...

public class SystemOutToSlf4j extends PrintStream {

  private static final PrintStream originalSystemOut = System.out;
  private static SystemOutToSlf4j systemOutToLogger;

  /**
   * Enable forwarding System.out.println calls to the logger if the stacktrace contains the class parameter
   * @param clazz
   */
  public static void enableForClass(Class clazz) {
    systemOutToLogger = new SystemOutToSlf4j(originalSystemOut, clazz.getName());
    System.setOut(systemOutToLogger);
  }


  /**
   * Enable forwarding System.out.println calls to the logger if the stacktrace contains the package parameter
   * @param packageToLog
   */
  public static void enableForPackage(String packageToLog) {
    systemOutToLogger = new SystemOutToSlf4j(originalSystemOut, packageToLog);
    System.setOut(systemOutToLogger);
  }

  /**
   * Disable forwarding to the logger resetting the standard output to the console
   */
  public static void disable() {
    System.setOut(originalSystemOut);
    systemOutToLogger = null;
  }

  private String packageOrClassToLog;

  private SystemOutToSlf4j(PrintStream original, String packageOrClassToLog) {
    super(original);
    this.packageOrClassToLog = packageOrClassToLog;
  }

  @Override
  public void println(String line) {
    StackTraceElement[] stack = Thread.currentThread().getStackTrace();
    StackTraceElement caller = findCallerToLog(stack);
    if (caller == null) {
      super.println(line);
      return;
    }

    org.slf4j.Logger log = org.slf4j.LoggerFactory.getLogger(caller.getClass());
    log.info(line);
  }

  public StackTraceElement findCallerToLog(StackTraceElement[] stack) {
    for (StackTraceElement element : stack) {
      if (element.getClassName().startsWith(packageOrClassToLog))
        return element;
    }

    return null;
  }

}

答案 2

我的建议是如果可能的话进行重构。有关可能的解决方案,请查看这些类似问题

log4j redirect stdout to DailyRollingFileAppender

将 System.out.println 重定向到 Log4J,同时保留类名信息


推荐