Javaロギング出力を1行に表示するにはどうすればよいですか?


124

現時点では、デフォルトのエントリは次のようになります。

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

これを行うにはどうすればよいですか?

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

明確化私はjava.util.loggingを使用しています

回答:


81

Java 7以降、java.util.logging.SimpleFormatterはシステムプロパティからのフォーマットの取得をサポートしているため、次のようなものをJVMコマンドラインに追加すると、1行に出力されます。

-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の時点で真。これについては、Java6のドキュメントでは言及されていません。
jbruni

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

1
...またはでlogging.properties、@ BrunoEberhardの提案に基づいた適応と短縮されたロガー名: java.util.logging.SimpleFormatter.format=%1$tF %1$tT %4$.1s %2$s %5$s%6$s%n
Kariem

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になる場合、つまり、プロパティファイルの設定でフォーマッタクラスが見つからなかった場合は、2つのことを実行できます。1。クラスがパッケージ内にあることを確認し、フォーマッタプロパティをそのクラスに設定します。そして、パッケージ2.クラスに「MyFormatter」のような一意の名前が付いていることを確認してください。最後に、すべてのハンドラーに目的のフォーマッターがあることを確認してください(java.util.logging.ConsoleHandler.formatter = my.package.MyFormatterおよび、java.util)。 .logging.FileHandler.formatter = my.package.MyFormatter)
Shaybc

共有いただきありがとうございます。別の質問答えて書いたフォーマッターを共有しました。それは私にとって大丈夫です。他の開発者がコードをより小さく、より速くするために思いつくトリックを見るのが好きです。
ライアン

1
@ondra-Žižka価値があるので、Googleコードリポジトリにパッチを送信しました。
ライアン14

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が作成される前に設定する必要があることに注意してください-たとえば、グローバルロガーのハンドラーを取得する前。
Sheepy

5
これは、JVMに他の起動パラメータを追加せずに、ログを強制的に1行に書き出す最も速い方法です。ただし、コードで 'new SimpleFormatter()'を呼び出す前に、必ずこのプロパティを設定してください。
サルバドールバレンシア

36

Obediah Staneが言ったように、独自のformatメソッドを作成する必要があります。しかし、いくつか変更します。

  • からFormatterではなく、から直接派生したサブクラスを作成しますSimpleFormatterSimpleFormatterもう追加には何もしています。

  • 新しいDateオブジェクトの作成には注意してください。の日付を表すようにしてくださいLogRecordDateデフォルトのコンストラクタで新しいを作成する場合、が作成された日付ではなく、がFormatterプロセスを実行した日付と時刻を表します。LogRecordLogRecord

以下のクラスは、することができるフォーマッタとして使用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になる場合、つまり、プロパティファイルの設定でフォーマッタクラスが見つからなかった場合は、2つのことを実行できます。1。クラスがパッケージ内にあることを確認し、フォーマッタプロパティをそのクラスに設定します。そして、パッケージ2.クラスに「MyFormatter」のような一意の名前が付いていることを確認してください。最後に、すべてのハンドラーに目的のフォーマッターがあることを確認してください(java.util.logging.ConsoleHandler.formatter = my.package.MyFormatterおよび、java.util)。 .logging.FileHandler.formatter = my.package.MyFormatter)
Shaybc

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で機能します。Java6にはこの機能がありません。
ジョシュアデイビス

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.アンデルセン

-2

tomcatを使用してWebアプリケーションにログインする場合は、以下を追加します。

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

VM引数について


-3

java.util.loggingを使用している場合、コンテンツをログに記録するための構成ファイルがあります(プログラムによる構成を使用している場合を除く)。したがって、オプションは
1)改行を削除するポストプロセッサーを実行する
2)ログ設定を変更し、そこから改行を削除することです。アプリケーション(サーバー)を再起動してください。

弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.