在尝试生成具有超过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
。
有解决方案吗?
答案 0 :(得分:1)
您已经回答了自己的问题。当生成的工作簿存储在内存中时,您的源似乎正在流式传输(rs.next())。电子表格对象中有很多内部对象引用,这显然导致复杂的垃圾收集器开销。
为避免这种情况,请使用XSSFWorkbook(SXSSFWorkbook)的流版本。
但是要回到为什么您可能会问这个问题/不想串流的原因: 您可能希望将所有寄存器都保留在内存中,以便可以更新标头信息(打印寄存器的总数,平均值等)。我很害怕这种操作无法无限扩展,您可以通过两次遍历源代码来实现此目的(首先累积所有次要信息,例如平均值,最大值或最小值,然后打印内容),或通过打印所有这些信息都在辅助页面上,您可以在第一页面中使用“ =“-formula”对其进行引用,因此看起来该信息位于页面顶部。