Java의 System.out에서 테이블 형식으로 출력


답변:


145

사용 System.out.format. 다음과 같이 필드 길이를 설정할 수 있습니다.

System.out.format("%32s%10d%16s", string1, int1, string2);

이렇게 string1하면 int1,, 및 string2각각 32, 10 및 16자를 채 웁니다.

대한 javadocs를 보라 java.util.Formatter구문에 대한 자세한 내용은 (대한 System.out.format사용 Formatter내부적으로).


1
내가 필요한 것만 감사합니다! 다른 사람이 문제를 발견 할 경우를 대비하여 매번 새 줄을 생성하기 위해 끝에 \ n을 추가해야합니다.
Chris

19
또는 더 나은 방법은 %n-자동으로 시스템의 기본 라인 종결 자로 변환됩니다.
Michael Myers

1
문자열 테이블에서 왼쪽 정렬을 어떻게 할 수 있습니까?
Mike

16
왼쪽 정렬 : System.out.format ( "%-2s % 10d % -16s", string1, int1, string2);
Mike

1
중앙 정렬하려면 어떻게해야합니까?
pratnala

36

j-text-utils 를 사용 하면 다음과 같은 테이블을 콘솔에 인쇄 할 수 있습니다. 여기에 이미지 설명 입력

다음과 같이 간단합니다.

TextTable tt = new TextTable(columnNames, data);                                                         
tt.printTable();   

API는 또한 정렬 및 행 번호 매기기를 허용합니다.


1
이것은 사용하기 다소 고통 스럽지만 문자열 형식을 지속적으로 조정하는 것보다 좋습니다. 그것은 더 나은 경우 것 : - 메이븐은 REPO에 소스 항아리를 발행 - 그것은 중앙과 같은 잘 알려진 REPO에 있었다
그렉 Chabala

15

훨씬 고급 테이블보기를 만들 수있는 프로젝트를 만들었습니다. 표 를 인쇄 하려면 표의 너비에 제한이 있습니다. 고객 송장 인쇄를 받기 위해 내 프로젝트 중 하나에 적용했습니다. 다음은 인쇄보기의 예입니다.

           PLATINUM COMPUTERS(PVT) LTD          
     NO 20/B, Main Street, Kandy, Sri Lanka.    
  Land: 812254630 Mob: 712205220 Fax: 812254639 

                CUSTOMER INVOICE                

+-----------------------+----------------------+
|INFO                   |CUSTOMER              |
+-----------------------+----------------------+
|DATE: 2015-9-8         |ModernTec Distributors|
|TIME: 10:53:AM         |MOB: +94719530398     |
|BILL NO: 12            |ADDRES: No 25, Main St|
|INVOICE NO: 458-80-108 |reet, Kandy.          |
+-----------------------+----------------------+
|                SELLING DETAILS               |
+-----------------+---------+-----+------------+
|ITEM             | PRICE($)|  QTY|       VALUE|
+-----------------+---------+-----+------------+
|Optical mouse    |   120.00|   20|     2400.00|
|Gaming keyboard  |   550.00|   30|    16500.00|
|320GB SATA HDD   |   220.00|   32|     7040.00|
|500GB SATA HDD   |   274.00|   13|     3562.00|
|1TB SATA HDD     |   437.00|   11|     4807.00|
|RE-DVD ROM       |   144.00|   29|     4176.00|
|DDR3 4GB RAM     |   143.00|   13|     1859.00|
|Blu-ray DVD      |    94.00|   28|     2632.00|
|WR-DVD           |   122.00|   34|     4148.00|
|Adapter          |   543.00|   28|    15204.00|
+-----------------+---------+-----+------------+
|               RETURNING DETAILS              |
+-----------------+---------+-----+------------+
|ITEM             | PRICE($)|  QTY|       VALUE|
+-----------------+---------+-----+------------+
|320GB SATA HDD   |   220.00|    4|      880.00|
|WR-DVD           |   122.00|    7|      854.00|
|1TB SATA HDD     |   437.00|    7|     3059.00|
|RE-DVD ROM       |   144.00|    4|      576.00|
|Gaming keyboard  |   550.00|    6|     3300.00|
|DDR3 4GB RAM     |   143.00|    7|     1001.00|
+-----------------+---------+-----+------------+
                              GROSS   59,928.00 
                       DISCOUNT(5%)    2,996.40 
                             RETURN    9,670.00 
                            PAYABLE   47,261.60 
                               CASH   20,000.00 
                             CHEQUE   15,000.00 
                    CREDIT(BALANCE)   12,261.60 






  ---------------------   --------------------- 
     CASH COLLECTOR         GOODS RECEIVED BY   

             soulution by clough.com            

이것은 위의 인쇄보기에 대한 코드이며 여기 에서 라이브러리 (Wagu)를 찾을 수 있습니다 .


1
@CLOUGH 씨, 귀하의 답변에 정말 감사드립니다. 인보이스를 생성하는 좋은 방법입니다. 나도 프로젝트를 진행하고 있지만 내 인보이스의 RETURNING DETAIL 섹션을 원하지 않습니다 ...이 부분을 도와 주실 수 있나요?.이 섹션을 잘라낼 수 없습니다
Rafi Abro

1
@LZH-예, 잘 작동합니다. 그러나 TextArea의 서체를 Monospace 로 변경해야합니다 .
CLOUGH

1
Maven 중앙 저장소에 게시됩니까?
Hubbitus

1
그렇게 할 계획입니까?
Hubbitus

1
@Hubbitus, 사실 저는이 라이브러리의 논리를 다시 프로그래밍 할 계획입니다. 결국 나는 이것을 maven에 게시 할 것입니다.
CLOUGH

15

나는 대답에 매우 늦을 수 있지만 여기에 간단하고 일반적인 해결책이 있습니다.

import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.UUID;

public class TableGenerator {

    private int PADDING_SIZE = 2;
    private String NEW_LINE = "\n";
    private String TABLE_JOINT_SYMBOL = "+";
    private String TABLE_V_SPLIT_SYMBOL = "|";
    private String TABLE_H_SPLIT_SYMBOL = "-";

    public String generateTable(List<String> headersList, List<List<String>> rowsList,int... overRiddenHeaderHeight)
    {
        StringBuilder stringBuilder = new StringBuilder();

        int rowHeight = overRiddenHeaderHeight.length > 0 ? overRiddenHeaderHeight[0] : 1; 

        Map<Integer,Integer> columnMaxWidthMapping = getMaximumWidhtofTable(headersList, rowsList);

        stringBuilder.append(NEW_LINE);
        stringBuilder.append(NEW_LINE);
        createRowLine(stringBuilder, headersList.size(), columnMaxWidthMapping);
        stringBuilder.append(NEW_LINE);


        for (int headerIndex = 0; headerIndex < headersList.size(); headerIndex++) {
            fillCell(stringBuilder, headersList.get(headerIndex), headerIndex, columnMaxWidthMapping);
        }

        stringBuilder.append(NEW_LINE);

        createRowLine(stringBuilder, headersList.size(), columnMaxWidthMapping);


        for (List<String> row : rowsList) {

            for (int i = 0; i < rowHeight; i++) {
                stringBuilder.append(NEW_LINE);
            }

            for (int cellIndex = 0; cellIndex < row.size(); cellIndex++) {
                fillCell(stringBuilder, row.get(cellIndex), cellIndex, columnMaxWidthMapping);
            }

        }

        stringBuilder.append(NEW_LINE);
        createRowLine(stringBuilder, headersList.size(), columnMaxWidthMapping);
        stringBuilder.append(NEW_LINE);
        stringBuilder.append(NEW_LINE);

        return stringBuilder.toString();
    }

    private void fillSpace(StringBuilder stringBuilder, int length)
    {
        for (int i = 0; i < length; i++) {
            stringBuilder.append(" ");
        }
    }

    private void createRowLine(StringBuilder stringBuilder,int headersListSize, Map<Integer,Integer> columnMaxWidthMapping)
    {
        for (int i = 0; i < headersListSize; i++) {
            if(i == 0)
            {
                stringBuilder.append(TABLE_JOINT_SYMBOL);   
            }

            for (int j = 0; j < columnMaxWidthMapping.get(i) + PADDING_SIZE * 2 ; j++) {
                stringBuilder.append(TABLE_H_SPLIT_SYMBOL);
            }
            stringBuilder.append(TABLE_JOINT_SYMBOL);
        }
    }


    private Map<Integer,Integer> getMaximumWidhtofTable(List<String> headersList, List<List<String>> rowsList)
    {
        Map<Integer,Integer> columnMaxWidthMapping = new HashMap<>();

        for (int columnIndex = 0; columnIndex < headersList.size(); columnIndex++) {
            columnMaxWidthMapping.put(columnIndex, 0);
        }

        for (int columnIndex = 0; columnIndex < headersList.size(); columnIndex++) {

            if(headersList.get(columnIndex).length() > columnMaxWidthMapping.get(columnIndex))
            {
                columnMaxWidthMapping.put(columnIndex, headersList.get(columnIndex).length());
            }
        }


        for (List<String> row : rowsList) {

            for (int columnIndex = 0; columnIndex < row.size(); columnIndex++) {

                if(row.get(columnIndex).length() > columnMaxWidthMapping.get(columnIndex))
                {
                    columnMaxWidthMapping.put(columnIndex, row.get(columnIndex).length());
                }
            }
        }

        for (int columnIndex = 0; columnIndex < headersList.size(); columnIndex++) {

            if(columnMaxWidthMapping.get(columnIndex) % 2 != 0)
            {
                columnMaxWidthMapping.put(columnIndex, columnMaxWidthMapping.get(columnIndex) + 1);
            }
        }


        return columnMaxWidthMapping;
    }

    private int getOptimumCellPadding(int cellIndex,int datalength,Map<Integer,Integer> columnMaxWidthMapping,int cellPaddingSize)
    {
        if(datalength % 2 != 0)
        {
            datalength++;
        }

        if(datalength < columnMaxWidthMapping.get(cellIndex))
        {
            cellPaddingSize = cellPaddingSize + (columnMaxWidthMapping.get(cellIndex) - datalength) / 2;
        }

        return cellPaddingSize;
    }

    private void fillCell(StringBuilder stringBuilder,String cell,int cellIndex,Map<Integer,Integer> columnMaxWidthMapping)
    {

        int cellPaddingSize = getOptimumCellPadding(cellIndex, cell.length(), columnMaxWidthMapping, PADDING_SIZE);

        if(cellIndex == 0)
        {
            stringBuilder.append(TABLE_V_SPLIT_SYMBOL); 
        }

        fillSpace(stringBuilder, cellPaddingSize);
        stringBuilder.append(cell);
        if(cell.length() % 2 != 0)
        {
            stringBuilder.append(" ");
        }

        fillSpace(stringBuilder, cellPaddingSize);

        stringBuilder.append(TABLE_V_SPLIT_SYMBOL); 

    }

    public static void main(String[] args) {
        TableGenerator tableGenerator = new TableGenerator();

        List<String> headersList = new ArrayList<>(); 
        headersList.add("Id");
        headersList.add("F-Name");
        headersList.add("L-Name");
        headersList.add("Email");

        List<List<String>> rowsList = new ArrayList<>();

        for (int i = 0; i < 5; i++) {
            List<String> row = new ArrayList<>(); 
            row.add(UUID.randomUUID().toString());
            row.add(UUID.randomUUID().toString());
            row.add(UUID.randomUUID().toString());
            row.add(UUID.randomUUID().toString());

            rowsList.add(row);
        }

        System.out.println(tableGenerator.generateTable(headersList, rowsList));
    }
}

이런 종류의 출력으로

+----------------------------------------+----------------------------------------+----------------------------------------+----------------------------------------+
|                   Id                   |                F-Name                  |                 L-Name                 |                  Email                 |
+----------------------------------------+----------------------------------------+----------------------------------------+----------------------------------------+
|  70a56f25-d42a-499c-83ac-50188c45a0ac  |  aa04285e-c135-46e2-9f90-988bf7796cd0  |  ac495ba7-d3c7-463c-8c24-9ffde67324bc  |  f6b5851b-41e0-4a4e-a237-74f8e0bff9ab  |
|  6de181ca-919a-4425-a753-78d2de1038ef  |  c4ba5771-ccee-416e-aebd-ef94b07f4fa2  |  365980cb-e23a-4513-a895-77658f130135  |  69e01da1-078e-4934-afb0-5afd6ee166ac  |
|  f3285f33-5083-4881-a8b4-c8ae10372a6c  |  46df25ed-fa0f-42a4-9181-a0528bc593f6  |  d24016bf-a03f-424d-9a8f-9a7b7388fd85  |  4b976794-aac1-441e-8bd2-78f5ccbbd653  |
|  ab799acb-a582-45e7-ba2f-806948967e6c  |  d019438d-0a75-48bc-977b-9560de4e033e  |  8cb2ad11-978b-4a67-a87e-439d0a21ef99  |  2f2d9a39-9d95-4a5a-993f-ceedd5ff9953  |
|  78a68c0a-a824-42e8-b8a8-3bdd8a89e773  |  0f030c1b-2069-4c1a-bf7d-f23d1e291d2a  |  7f647cb4-a22e-46d2-8c96-0c09981773b1  |  0bc944ef-c1a7-4dd1-9eef-915712035a74  |
+----------------------------------------+----------------------------------------+----------------------------------------+----------------------------------------+

필요한 유일한 가져 오기는 다음과 같습니다. import java.util.HashMap; import java.util.List; import java.util.Map;
Jake Hm

실제 예제와 함께 public static void main을 제공 할 수 있다면 나쁘지 않을 것입니다.
mmm

1
이 솔루션은 "간단한"호출 할 수 없습니다
Gauraang Khurana

11

이것 좀 봐. 저자는 타사 라이브러리가 필요하지 않은 간단하지만 우아한 솔루션을 제공합니다. http://www.ksmpartners.com/2013/08/nicely-formatted-tabular-output-in-java/

TableBuilder 및 샘플 출력의 예


이 솔루션은 타사 라이브러리 인 Apache Lang을 매우 잘 사용하지만 여전히 내 필요에 가장 적합한 솔루션입니다. 1+
Spenhouet 2015 년

인터넷에서 길잃은 코드를 복사하는 것은 실제로 타사 라이브러리보다 약간 더 나쁩니다. 다른 사람이 재사용 할 수 있도록 패키지화하는 데 아무도 신경 쓰지 않은 라이브러리입니다.
Greg Chabala

7

대부분의 솔루션이 약간 구식이기 때문에 이미 maven ( ) 에서 사용할 수 있고 매우 복잡한 구성을 생성 할 수있는 asciitable 을 제안 할 수도 de.vandermeer:asciitable:0.3.2있습니다.

기능 (오프 사이트) :

  • 규칙 및 내용, 정렬, 형식, 패딩, 여백 및 프레임에 대한 약간의 유연성이있는 텍스트 테이블 :
  • 다양한 형식 (문자열, 텍스트 공급자, 렌더링 공급자, ST, 클러스터)에서 필요한만큼 자주 텍스트를 추가합니다.
  • 과도한 공백 (표 작성기, 추가 공백, 캐리지 리턴 및 줄 바꿈 조합)을 모두 제거합니다.
  • 6 가지 텍스트 정렬 : 왼쪽, 오른쪽, 가운데 맞춤, 양쪽 맞춤, 마지막 줄 왼쪽 맞춤, 마지막 줄 오른쪽 맞춤,
  • 유연한 너비, 텍스트 용으로 설정되고 렌더링을 위해 다양한 방법으로 계산 됨
  • 왼쪽 및 오른쪽 패딩을위한 패딩 문자 (별도 구성 가능)
  • 상단 및 하단 패딩을위한 패딩 문자 (별도 구성 가능)
  • 그리드 그리기를위한 몇 가지 옵션
  • 스타일이 다른 규칙 (사용 된 그리드 테마에서 지원하는대로 : 보통, 밝게, 강함, 무겁게)
  • 프레임 외부의 위쪽 / 아래쪽 / 왼쪽 / 오른쪽 여백
  • 추가 프로세스에 적합한 생성 된 텍스트로 문자 변환 (예 : LaTeX 및 HTML)

그리고 사용법은 여전히 ​​쉽습니다.

AsciiTable at = new AsciiTable();

at.addRule();
at.addRow("row 1 col 1", "row 1 col 2");
at.addRule();
at.addRow("row 2 col 1", "row 2 col 2");
at.addRule();

System.out.println(at.render()); // Finally, print the table to standard out.

2
니스와 잘 문서화 라이브러리
데이비드 Lilljegren

5
public class Main {
 public static void main(String args[]) {
   String format = "|%1$-10s|%2$-10s|%3$-20s|\n";
   System.out.format(format, "A", "AA", "AAA");
   System.out.format(format, "B", "", "BBBBB");
   System.out.format(format, "C", "CCCCC", "CCCCCCCC");

   String ex[] = { "E", "EEEEEEEEEE", "E" };

   System.out.format(String.format(format, (Object[]) ex));
 }
}

입력 크기의 차이가 출력에 영향을주지 않음


당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.