Jak sprawić, by dane wyjściowe logowania Java pojawiały się w jednym wierszu?


124

W tej chwili domyślny wpis wygląda mniej więcej tak:

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

Jak mam to zrobić?

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

Wyjaśnienie Używam java.util.logging

Odpowiedzi:


81

Począwszy od Java 7, java.util.logging.SimpleFormatter obsługuje pobieranie formatu z właściwości systemowej, więc dodanie czegoś takiego do wiersza poleceń maszyny JVM spowoduje wydrukowanie go w jednym wierszu:

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

Alternatywnie możesz również dodać to do 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
Prawda od Java7. Nie ma o tym wzmianki w dokumentacji Java6.
jbruni

1
Również w IDEA działa z podwójnymi cudzysłowami. @jbruni To nie działa w Javie6.
Joshua Davis

1
Działa z pojedynczymi cudzysłowami w Eclipse.
bogaty

1
Zamiast% 1 $ tY-% 1 $ tm-% 1 $ td% 1 $ tH:% 1 $ tM:% 1 $ tS możesz napisać% 1 $ tF% 1 $ tT. Sprawia, że ​​jest trochę krótszy. Nie wiem, czy to jeszcze szybciej.
Bruno Eberhard

1
... lub in logging.properties, z adaptacją opartą na sugestii @BrunoEberhard i skróconą nazwą loggera: java.util.logging.SimpleFormatter.format=%1$tF %1$tT %4$.1s %2$s %5$s%6$s%n
Kariem 20.04.2015

73

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

Java 7 obsługuje właściwość ze java.util.Formatterskładnią łańcucha formatu.

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

Zobacz tutaj .

Moim ulubionym jest:

-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

co sprawia, że ​​dane wyjściowe wyglądają następująco:

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

2) Umieszczenie go w IDE

Środowiska IDE zazwyczaj pozwalają ustawić właściwości systemowe projektu. Np. W NetBeans, zamiast dodawać gdzieś -D ... = ..., dodaj właściwość w oknie dialogowym akcji, w postaci java.util.logging.SimpleFormatter.format=%1$tY-%1$tm-...- bez cudzysłowów. IDE powinno się dowiedzieć.

3) Przekazanie tego Mavenowi - Surefire

Dla Twojej wygody, oto jak umieścić to w 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) ręcznie

Mam bibliotekę z kilkoma java.util.loggingpowiązanymi klasami . Wśród nich jest SingleLineFormatter. Jar do pobrania tutaj .

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

lepsze zużycie pamięci niż inne rozwiązanie.
Tim Williscroft,

Próbowałem tego - ale to daje mi log w XML - co robię źle - stackoverflow.com/questions/16030578/ ...
user93353 Kwietnia

jeśli zmieni XML zamiast oczekiwanego formatu - oznacza to, że ustawienie w pliku właściwości nie znalazło Twojej klasy programu formatującego, możesz zrobić 2 rzeczy: 1. Upewnij się, że Twoja klasa jest w pakiecie i ustaw właściwość programu formatującego na tę klasę i pakiet 2. upewnij się, że Twoja klasa ma unikalną nazwę, taką jak „MyFormatter”, na koniec upewnij się, że wszystkie programy obsługi mają żądany program formatujący (java.util.logging.ConsoleHandler.formatter = my.package.MyFormatter, a także: java.util .logging.FileHandler.formatter = my.package.MyFormatter)
Shaybc

Dzięki za udostępnienie. Udostępniłem Formatter, który napisałem w odpowiedzi na inne pytanie . Jak dla mnie wykonał to dobrze. Lubię patrzeć na sztuczki, które wymyślają inni programiści, aby kod był mniejszy i szybszy.
Ryan,

1
@ ondra-Žižka Na ile to warte, właśnie wysłałem ci łatkę na repozytorium kodu Google.
Ryan,

62

Podobnie jak Tervor, ale lubię zmieniać właściwość w czasie wykonywania.

Zauważ, że należy to ustawić przed utworzeniem pierwszego SimpleFormatter - tak jak napisano w komentarzach.

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

6
Należy pamiętać, że należy to ustawić przed utworzeniem pierwszego SimpleFormatter - na przykład przed uzyskaniem globalnych programów obsługi rejestratora.
Sheepy

5
Jest to najszybszy sposób, aby wymusić zapisywanie dzienników w jednym wierszu bez dodawania innych parametrów startowych do maszyny JVM. Ale upewnij się, że ustawiłeś tę właściwość przed wywołaniem w kodzie funkcji „new SimpleFormatter ()”.
Salvador Valencia

36

Jak powiedział Obediah Stane, konieczne jest stworzenie własnej formatmetody. Ale zmieniłbym kilka rzeczy:

  • Utwórz podklasę pochodzącą bezpośrednio z Formatter, a nie z SimpleFormatter. Nie SimpleFormatterma już nic do dodania.

  • Uważaj przy tworzeniu nowego Dateobiektu! Powinieneś upewnić się, że reprezentujesz datę LogRecord. Podczas tworzenia nowego Datez domyślnym konstruktorem będzie on reprezentował datę i godzinę Formatterprzetwarzania LogRecord, a nie datę LogRecordutworzenia.

Następującej klasy można użyć jako programu formatującego w pliku Handler, który z kolei można dodać do Logger. Zauważ, że ignoruje wszystkie informacje o klasach i metodach dostępne w 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();
    }
}

Próbowałem tego - ale to daje mi log w XML - co robię źle - stackoverflow.com/questions/16030578/ ...
user93353 Kwietnia

jeśli zmieni XML zamiast oczekiwanego formatu - oznacza to, że ustawienie w pliku właściwości nie znalazło Twojej klasy programu formatującego, możesz zrobić 2 rzeczy: 1. Upewnij się, że Twoja klasa jest w pakiecie i ustaw właściwość programu formatującego na tę klasę i pakiet 2. upewnij się, że Twoja klasa ma unikalną nazwę, taką jak „MyFormatter”, na koniec upewnij się, że wszystkie programy obsługi mają żądany program formatujący (java.util.logging.ConsoleHandler.formatter = my.package.MyFormatter, a także: java.util .logging.FileHandler.formatter = my.package.MyFormatter)
Shaybc

10

Tego właśnie używam.

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

Otrzymasz coś takiego ...

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

Jak mogliśmy uzyskać informacje o wyjątku wyjściowym tego programu formatującego, takie jak jego ślad stosu?
fegemo

1
@fegemo Myślę, że możesz wydrukować starcktrace w ten sposób. ofNullable(record.getThrown()).ifPresent(v -> v.printStackTrace());tuż przed zwróceniem sformatowanej wiadomości.
Jin Kwon

7

Konfiguracja Eclipse

Na zrzucie ekranu w Eclipse wybierz „uruchom jako”, a następnie „Uruchom konfiguracje ...” i dodaj odpowiedź od Trevora Robinsona w podwójnych cudzysłowach zamiast cudzysłowów. Jeśli pominiesz podwójne cudzysłowy, otrzymasz błąd „nie można znaleźć lub załadować głównej klasy”.


2
Zgodnie z poprzednią odpowiedzią, działa to w Javie 7. Java 6 nie ma tej funkcji.
Joshua Davis

5

Znalazłem sposób, który działa. Możesz podklasować SimpleFormatter i przesłonić metodę formatowania

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

Nieco zaskoczony tym interfejsem API, pomyślałem, że po wyjęciu z pudełka zapewniono by większą funkcjonalność / elastyczność


Każdy powinien formatMessage(record)raczej używać niż record.getMessage(). Posiadacze miejsca nie ulegną zmianie.
Jin Kwon

-2

To rejestrowanie jest specyficzne dla twojej aplikacji i nie jest ogólną funkcją Java. Jakie aplikacje używasz?

Możliwe, że pochodzi z określonej biblioteki rejestrowania, której używasz w swoim własnym kodzie. Jeśli tak, podaj szczegółowe informacje, z którego korzystasz.


3
To nie jest jakaś aplikacja do losowego rejestrowania. Jest własnością javyjava.util.logging.Logger
André C. Andersen

-2

Jeśli logujesz się do aplikacji internetowej za pomocą tomcat, dodaj:

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

Na argumentach maszyny wirtualnej


-3

jeśli używasz java.util.logging, istnieje plik konfiguracyjny, który robi to w celu rejestrowania treści (chyba że używasz konfiguracji programowej). Masz więc opcje:
1) uruchom postprocesor, który usunie znaki końca wierszy
2) zmień konfigurację dziennika ORAZ usuń z niego znaki końca wierszy. Uruchom ponownie aplikację (serwer) i powinno być dobrze.

Korzystając z naszej strony potwierdzasz, że przeczytałeś(-aś) i rozumiesz nasze zasady używania plików cookie i zasady ochrony prywatności.
Licensed under cc by-sa 3.0 with attribution required.