如何使Java日志输出显示在一行上?


124

目前,默认条目如下所示:

Oct 12, 2008 9:45:18 AM myClassInfoHere
INFO: MyLogMessageHere

我该怎么做呢?

Oct 12, 2008 9:45:18 AM myClassInfoHere - INFO: MyLogMessageHere

澄清我正在使用java.util.logging

Answers:


81

从Java 7开始,java.util.logging.SimpleFormatter支持从系统属性获取其格式,因此在JVM命令行中添加类似内容将使其在一行上打印:

-Djava.util.logging.SimpleFormatter.format='%1$tY-%1$tm-%1$td %1$tH:%1$tM:%1$tS %4$s %2$s %5$s%6$s%n'

另外,您也可以将其添加到您的logger.properties

java.util.logging.SimpleFormatter.format='%1$tY-%1$tm-%1$td %1$tH:%1$tM:%1$tS %4$s %2$s %5$s%6$s%n'

4
从Java7开始为true。Java6文档中没有提及这一点。
jbruni 2012年

1
同样在IDEA中,可以使用双引号。@jbruni在Java6中不起作用。
约书亚·戴维斯

1
我可以在Eclipse中使用单引号。
丰富的

1
代替%1 $ tY-%1 $ tm-%1 $ td%1 $ tH:%1 $ tM:%1 $ tS,您可以编写%1 $ tF%1 $ tT。使它短一点。我不知道它是否更快。
Bruno Eberhard 2014年

1
...或中logging.properties,根据@BrunoEberhard的建议进行改编,并使用简短的记录器名称: java.util.logging.SimpleFormatter.format=%1$tF %1$tT %4$.1s %2$s %5$s%6$s%n
Kariem 2015年

73

1) -Djava.util.logging.SimpleFormatter.format

Java 7支持java.util.Formatter格式字符串语法的属性。

-Djava.util.logging.SimpleFormatter.format=... 

这里

我最喜欢的是:

-Djava.util.logging.SimpleFormatter.format=%1$tY-%1$tm-%1$td %1$tH:%1$tM:%1$tS %4$-6s %2$s %5$s%6$s%n

这使得输出像:

2014-09-02 16:44:57 SEVERE org.jboss.windup.util.ZipUtil unzip: Failed to load: foo.zip

2)放入IDE

IDE通常使您可以设置项目的系统属性。例如,在NetBeans中,不要在某处添加-D ... = ...,而应在操作对话框中以java.util.logging.SimpleFormatter.format=%1$tY-%1$tm-...- 的形式添加该属性,且不带引号。IDE应该弄清楚。

3)把它放到Maven-Surefire

为了您的方便,以下是将其放入Surefire的方法:

        <!-- Surefire -->
        <plugin>
            <groupId>org.apache.maven.plugins</groupId>
            <artifactId>maven-surefire-plugin</artifactId>
            <version>2.17</version>
            <configuration>
                <systemPropertyVariables>
                    <!-- Set JUL Formatting -->
                    <java.util.logging.SimpleFormatter.format>%1$tY-%1$tm-%1$td %1$tH:%1$tM:%1$tS %4$-6s %2$s %5$s%6$s%n</java.util.logging.SimpleFormatter.format>
                </systemPropertyVariables>
            </configuration>
        </plugin>

4)手工制作

我有一个很少有java.util.logging相关课程的图书馆。其中有SingleLineFormatter。可下载的jar 在这里

public class SingleLineFormatter extends Formatter {

  Date dat = new Date();
  private final static String format = "{0,date} {0,time}";
  private MessageFormat formatter;
  private Object args[] = new Object[1];

  // Line separator string.  This is the value of the line.separator
  // property at the moment that the SimpleFormatter was created.
  //private String lineSeparator = (String) java.security.AccessController.doPrivileged(
  //        new sun.security.action.GetPropertyAction("line.separator"));
  private String lineSeparator = "\n";

  /**
   * Format the given LogRecord.
   * @param record the log record to be formatted.
   * @return a formatted log record
   */
  public synchronized String format(LogRecord record) {

    StringBuilder sb = new StringBuilder();

    // Minimize memory allocations here.
    dat.setTime(record.getMillis());    
    args[0] = dat;


    // Date and time 
    StringBuffer text = new StringBuffer();
    if (formatter == null) {
      formatter = new MessageFormat(format);
    }
    formatter.format(args, text, null);
    sb.append(text);
    sb.append(" ");


    // Class name 
    if (record.getSourceClassName() != null) {
      sb.append(record.getSourceClassName());
    } else {
      sb.append(record.getLoggerName());
    }

    // Method name 
    if (record.getSourceMethodName() != null) {
      sb.append(" ");
      sb.append(record.getSourceMethodName());
    }
    sb.append(" - "); // lineSeparator



    String message = formatMessage(record);

    // Level
    sb.append(record.getLevel().getLocalizedName());
    sb.append(": ");

    // Indent - the more serious, the more indented.
    //sb.append( String.format("% ""s") );
    int iOffset = (1000 - record.getLevel().intValue()) / 100;
    for( int i = 0; i < iOffset;  i++ ){
      sb.append(" ");
    }


    sb.append(message);
    sb.append(lineSeparator);
    if (record.getThrown() != null) {
      try {
        StringWriter sw = new StringWriter();
        PrintWriter pw = new PrintWriter(sw);
        record.getThrown().printStackTrace(pw);
        pw.close();
        sb.append(sw.toString());
      } catch (Exception ex) {
      }
    }
    return sb.toString();
  }
}

比其他解决方案更好的内存消耗。
Tim Williscroft 2011年

我想这一点-但它给了我日志中的XML -我在做什么错- stackoverflow.com/questions/16030578/...
user93353

如果它变成XML而不是您期望的格式-这意味着属性文件中的设置未找到您的格式化程序类,则您可以做两件事:1.确保您的类在包中,然后将格式化程序属性设置为该类和包2.确保您的类具有唯一的名称,例如“ MyFormatter”,最后:确保所有处理程序都具有所需的格式化程序(java.util.logging.ConsoleHandler.formatter = my.package.MyFormatter以及java.util .logging.FileHandler.formatter = my.package.MyFormatter)
Shaybc 2014年

感谢分享。我分享了为回答另一个问题而写的格式化程序。它对我来说还不错。我喜欢看到其他开发人员提出的使代码更小,更快的技巧。
Ryan

1
@ondra-Žižka我刚刚为您发送了有关Google代码存储库的补丁,以说明其价值。
Ryan

61

与Tervor类似,但我想在运行时更改属性。

请注意,这需要在创建第一个SimpleFormatter之前进行设置-如注释中所写。

    System.setProperty("java.util.logging.SimpleFormatter.format", 
            "%1$tF %1$tT %4$s %2$s %5$s%6$s%n");

6
请注意,这需要在创建第一个SimpleFormatter之前进行设置-例如,在获取全局记录程序的处理程序之前。
羊皮的

5
这是强制将日志写出到一行而不向JVM添加任何其他启动参数的最快方法。但是,在代码中调用“ new SimpleFormatter()”之前,请确保设置此属性。
萨尔瓦多·瓦伦西亚

36

就像Obediah Stane所说的那样,有必要创建自己的format方法。但是我会改变一些事情:

  • 创建直接源自而Formatter不是的子类SimpleFormatter。该SimpleFormatter有什么可再补充。

  • 创建新Date对象时要小心!您应该确保代表的日期LogRecord。当创建一个新的Date默认构造,其所代表的日期和时间Formatter过程LogRecord的,不是日期LogRecord的创建。

下面的类可以被用作格式化器中一个Handler,这反过来又可以加入Logger。请注意,它会忽略中提供的所有类和方法信息LogRecord

import java.io.PrintWriter;
import java.io.StringWriter;
import java.util.Date;
import java.util.logging.Formatter;
import java.util.logging.LogRecord;

public final class LogFormatter extends Formatter {

    private static final String LINE_SEPARATOR = System.getProperty("line.separator");

    @Override
    public String format(LogRecord record) {
        StringBuilder sb = new StringBuilder();

        sb.append(new Date(record.getMillis()))
            .append(" ")
            .append(record.getLevel().getLocalizedName())
            .append(": ")
            .append(formatMessage(record))
            .append(LINE_SEPARATOR);

        if (record.getThrown() != null) {
            try {
                StringWriter sw = new StringWriter();
                PrintWriter pw = new PrintWriter(sw);
                record.getThrown().printStackTrace(pw);
                pw.close();
                sb.append(sw.toString());
            } catch (Exception ex) {
                // ignore
            }
        }

        return sb.toString();
    }
}

我想这一点-但它给了我日志中的XML -我在做什么错- stackoverflow.com/questions/16030578/...
user93353

如果它变成XML而不是您期望的格式-这意味着您的属性文件中的设置未找到您的格式化程序类,您可以做两件事:1.确保您的类在一个包中,并将formatter属性设置为该类和包2.确保您的类具有唯一的名称,例如“ MyFormatter”,最后:确保所有处理程序都具有所需的格式化程序(java.util.logging.ConsoleHandler.formatter = my.package.MyFormatter以及java.util (.logging.FileHandler.formatter = my.package.MyFormatter)
Shaybc 2014年

10

这就是我正在使用的。

public class VerySimpleFormatter extends Formatter {

    private static final String PATTERN = "yyyy-MM-dd'T'HH:mm:ss.SSSXXX";

    @Override
    public String format(final LogRecord record) {
        return String.format(
                "%1$s %2$-7s %3$s\n",
                new SimpleDateFormat(PATTERN).format(
                        new Date(record.getMillis())),
                record.getLevel().getName(), formatMessage(record));
    }
}

你会得到类似...

2016-08-19T17:43:14.295+09:00 INFO    Hey~
2016-08-19T17:43:16.068+09:00 SEVERE  Seriously?
2016-08-19T17:43:16.068+09:00 WARNING I'm warning you!!!

我们如何让此格式化程序输出异常信息,例如其堆栈跟踪?
fegemo

1
@fegemo我想您可以像这样打印starcktrace。ofNullable(record.getThrown()).ifPresent(v -> v.printStackTrace());在返回格式化的消息之前。
晋权

7

Eclipse配置

在每个屏幕截图中,在Eclipse中选择“运行方式”,然后选择“运行配置...”,并使用双引号(而不是引号)添加Trevor Robinson的答案。如果您错过了双引号,则会收到“找不到或加载主类”错误。


2
根据先前的答案,此方法在Java 7中有效。Java 6没有此功能。
约书亚·戴维斯

5

我想出了一种可行的方法。您可以继承SimpleFormatter并重写format方法

    public String format(LogRecord record) {
        return new java.util.Date() + " " + record.getLevel() + " " + record.getMessage() + "\r\n";
    }

对此API感到有些惊讶,我本来以为会提供更多的功能/灵活性


任何人都应该使用formatMessage(record)而不是record.getMessage()。占位符不会改变。
金权

-2

此日志记录特定于您的应用程序,而不是常规的Java功能。您正在运行什么应用程序?

这可能是由于您在自己的代码中使用的特定日志记录库产生的。如果是这样,请张贴您正在使用的详细信息。


3
这不是一些随机记录器应用程序。这是Java的自己java.util.logging.Logger
安德烈C.安德森


-3

如果您使用的是java.util.logging,则有一个配置文件正在执行此操作以记录内容(除非您使用编程配置)。因此,您的选择是
1)运行后处理器以删除换行符
2)更改日志配置并从中删除换行符。重新启动您的应用程序(服务器),您应该会很好。

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.