Apache POI autoSizeColumn()表现得很奇怪

时间:2015-02-17 14:46:00

标签: java excel apache-poi

请在标记为重复之前阅读完整。我已经看过所有相关/类似的问题,我想我在这里发生了一些奇怪的事情。

我使用Apache POI版本3.9生成4张工作簿。每次我填写表格时,我会自动调整列数。这是代码:

for (String alias : map.keySet()) {
    Sheet sheet = workbook.createSheet(getSheetNameForAlias(alias));
    List<Object[]> resultList = (List<Object[]>) map.get(alias);
    Collections.sort(resultList.subList(1, resultList.size()), getComparator(alias));
    int rownum = 0;
    for (Object[] array : resultList) {
        rownum = populateRow(cellStyles, sheet, rownum, array, false);
    }

    for(int j=0; j< resultList.get(0).length; j++){
        sheet.autoSizeColumn(j, false);
    }
}

populateRow函数是:

private int populateRow(CellStyle[] cellStyles, Sheet sheet, int rownum, Object[] dataArr, boolean doAutoSizing) {

    if (logger.isDebugEnabled()) {
        logger.debug("Populating row ... @{} : {} ", sheet.getSheetName(), rownum);
    }

    int cellnum;
    Row row = sheet.createRow(rownum++);
    cellnum = 0;
    if (ArrayUtils.isEmpty(dataArr)) {
        logger.error("No data found for row ... @{} : {} ", sheet.getSheetName(), rownum);
        return rownum;
    }
    for (Object obj : dataArr) {
        Cell cell = row.createCell(cellnum++);

        // get the type of the data inserted in the cell.
        // Accordingly set the cell value
        String data = String.valueOf(obj);

        if (StringUtils.isBlank(data) || "null".equals(data)) {
            cell.setCellValue(StringUtils.EMPTY);
        } else {
            if (NumberUtils.isNumber(data)) {

                Double dbVal = NumberUtils.createDouble(data);

                // Millions separator for Numeric fields
                if (dbVal == Math.ceil(dbVal)) {
                    cell.setCellStyle(cellStyles[0]);
                } else {
                    cell.setCellStyle(cellStyles[1]);
                }
                cell.setCellType(Cell.CELL_TYPE_NUMERIC); // for numeric , set cell type as numeric
                cell.setCellValue(dbVal);
            } else {
                if(obj instanceof java.sql.Timestamp || obj instanceof java.sql.Date){
                  cell.setCellStyle(cellStyles[2]);
             }
                cell.setCellValue(data);
            }
            if (doAutoSizing) {
                sheet.autoSizeColumn(cellnum - 1);
            }
            data = null;
        }
        obj = null;
    }
    return rownum;
}

奇怪的是,四张纸中的一列的列调整不正确。

我尝试用不同的方法重新编写auto size逻辑并发现该表格,如果我尝试sheet.getRow(0)我得到了这个错误的行为,它会给出out null指定该行没有数据,但生成的excel不能很好地保存,所有四个表都有适当的数据。并且在每种情况下,第一行始终是标题,因此即使下面没有数据也不能为空。

我不知道自己错过了什么,但我试过并发现它不是JVM的字体问题。

如果您需要其他信息,请与我联系。

编辑:观察到自动调整大小对于包含100行以上的工作表不正确。我不知道如何将它与某些逻辑联系起来。这是真正的WEIRD。

编辑2:我找到了Axel在寻找其他东西时给出的理由的官方参考。这确实足以让我感到困惑。请阅读以下内容 link

1 个答案:

答案 0 :(得分:1)

由于您说它最多可以运行100行,因此看起来您使用的是SXSSFWorkbook,而不是XSSFWorkbook。然后必须预期该行为,因为在内存中只保留固定数量的行,并且仅考虑这些行。所以,如果我是对的,你将不得不切换到使用XSSFWorkbook(假设这在处理大量行时不会导致OOM)。