如何让 java 日志输出显示在一行上?

目前,默认条目大致如下:

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

我怎么才能让它这么做?

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

澄清我使用的是 java.util.log

122381 次浏览

这个日志记录是特定于您的应用程序的,而不是一般的 Java 特性?

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

如果您使用的是 java.util.log,那么有一个配置文件正在对内容进行日志记录(除非您使用的是编程配置)。所以,你的选择是
1)运行后处理程序,删除换行符
2)更改日志配置并删除断行。重新启动您的应用程序(服务器) ,您应该是良好的。

我已经找到了一种可行的方法。您可以继承 SimpleFormatter 并覆盖 format 方法

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

对于这个 API 我有点惊讶,我原以为会提供更多的功能/灵活性

就像 Obediah Stane 说的,创建自己的 format方法是必要的,但是我会改变一些东西:

  • 创建一个直接从 Formatter派生的子类,而不是从 SimpleFormatter派生的子类。

  • 小心创建一个新的 Date对象!你应该确保代表日期的 LogRecord。当使用缺省构造函数创建一个新的 Date时,它代表的是 Formatter处理 LogRecord的日期和时间,而不是 LogRecord创建的日期。

下面的类可以是 Handler中的 用作格式化程序,而 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();
}
}

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

Java7支持使用 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。 可下载的罐子。

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

从 Java 7开始,Loggs.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'

Eclipse config

根据屏幕截图,在 Eclipse 中选择“ Run as”,然后选择“ Run Configuration...”,并用双引号代替引号添加 Trevor Robinson 的答案。如果您错过了双引号,您将得到“无法找到或加载主类”错误。

类似于 Tervor,但我喜欢在运行时更改属性。

注意,这需要在创建第一个 SimpleFormatter 之前进行设置——正如在注释中所写的那样。

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

我用的就是这个。

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!!!

如果您使用 tomcat 登录 Web 应用程序,请添加:

-Djava.util.logging.ConsoleHandler.formatter = org.apache.juli.OneLineFormatter

关于 VM 参数