线程“主”中的POI异常java.lang.OutOfMemoryError:超出了GC开销限制

时间:2018-08-09 13:57:16

标签: java garbage-collection apache-poi

在尝试生成具有超过250万行的XLSX文件时,使用XSSFWorkbook时,我一直得到Exception in thread "main" java.lang.OutOfMemoryError: GC overhead limit exceeded

此外,我添加了JVM参数 -Xms1024M -Xmx6144 ,但没有成功。

while (rs.next()) {
    row = spreadsheet.createRow(rowID);
    for (int column = 0; column < numberOfColumns; column++) {
        value = (String) rs.getString(column + 1);
        cell = row.createCell(column);
        cell.setCellStyle(style);
        spreadsheet.setColumnWidth(column, COLUMN_WIDTH);
        formatedValue = Engine.formatInput(colNames.get(column),value);
        if (formatedValue instanceof Calendar) {
            cell.setCellStyle(dateCellStyle);
            cell.setCellValue((Calendar) formatedValue);
        } else {
            if (formatedValue instanceof Double) {
                cell.setCellValue((Double) formatedValue);
            } else {
                if (formatedValue instanceof Integer) {
                    cell.setCellValue((Integer) formatedValue);
                } else {
                    if (formatedValue instanceof String) {
                        cell.setCellValue((String) formatedValue);
                    } else {
                        /*
                         * Unreachable.
                         */
                        cell.setCellValue(value);
                    }
                }
            }
        }
    }
    rowID++;
    if (this.isMaxSpreadsheetRowsReached(rowID)) {
        newSpreadSheet(rsmd, numberOfColumns, styleEntete);
        rowID = 1;
    }
}

该异常发生在内部循环中。

一种解决方法是使用SXSSFWorkbook,但我仍然想使用XSSFWorkbook

有解决方案吗?

1 个答案:

答案 0 :(得分:1)

您已经回答了自己的问题。当生成的工作簿存储在内存中时,您的源似乎正在流式传输(rs.next())。电子表格对象中有很多内部对象引用,这显然导致复杂的垃圾收集器开销。

为避免这种情况,请使用XSSFWorkbook(SXSSFWorkbook)的流版本。

但是要回到为什么您可能会问这个问题/不想串流的原因: 您可能希望将所有寄存器都保留在内存中,以便可以更新标头信息(打印寄存器的总数,平均值等)。我很害怕这种操作无法无限扩展,您可以通过两次遍历源代码来实现此目的(首先累积所有次要信息,例如平均值,最大值或最小值,然后打印内容),或通过打印所有这些信息都在辅助页面上,您可以在第一页面中使用“ =“-formula”对其进行引用,因此看起来该信息位于页面顶部。