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을 사용하고 있습니다.

답변:


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에서 사실입니다. 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를 작성할 수 있습니다. 조금 더 짧게 만듭니다. 더 빠른지 모르겠습니다.
브루노 에버

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. 여기에서 다운로드 할 수있는 항아리 .

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 년

공유해 주셔서 감사합니다. 다른 질문대한 답변으로 작성한 Formatter를 공유했습니다 . 그것은 나를 위해 잘 수행되었습니다. 다른 개발자가 코드를 더 작고 빠르게 만들기 위해 고안 한 트릭을 보는 것이 좋습니다.
라이언

1
@ ondra-Žižka Google 코드 저장소에 패치를 보냈습니다.
라이언

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에 다른 시작 매개 변수를 추가하지 않고 로그를 한 줄에 기록하도록하는 가장 빠른 방법입니다. 그러나 코드에서 'new SimpleFormatter ()'를 호출하기 전에이 속성을 설정해야합니다.
Salvador Valencia

36

Obediah Stane이 말했듯이, 자신 만의 format방법 을 만들어야합니다 . 하지만 몇 가지를 변경하겠습니다.

  • 에서 Formatter가 아니라에서 직접 파생 된 하위 클래스를 만듭니다 SimpleFormatter. 은 SimpleFormatter더 이상 추가 아무 상관이 없습니다.

  • Date개체 를 만들 때주의하십시오 ! 의 날짜를 표시해야합니다 LogRecord. Date기본 생성자 로 새 파일 을 만들 때이 생성 된 날짜가 아니라를 Formatter처리하는 날짜와 시간을 나타냅니다 .LogRecordLogRecord

다음 클래스는 할 수 포맷으로 사용 A의 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. 클래스가 패키지에 있는지 확인하고 포맷터 속성을 해당 클래스로 설정합니다. 그리고 패키지 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());형식화 된 메시지를 반환하기 직전에.
Jin Kwon

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.util.logging.Logger
앙드레 C. 앤더슨

-2

tomcat을 사용하여 웹 애플리케이션에 로그인하는 경우 다음을 추가하십시오.

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