使用XSSFWorkbook
时,当我试图生成超过250万行的XLSX文件时,我总是得到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;
}
}
异常发生在内部for循环中。
一种解决方法是使用SXSSFWorkbook
,但我仍然想使用XSSFWorkbook
。
对此有解决方案吗?
发布于 2018-08-09 22:43:24
你已经回答了你自己的问题。您的源似乎是流式传输的(rs.next()),而生成的工作簿存储在内存中。在电子表格对象中有许多内部对象引用,这显然会导致复杂的垃圾收集器开销。
要避免这种情况,请使用XSSFWorkbook (SXSSFWorkbook)的流版本。
但回到您可能会问这个问题/不想流式传输它的原因:您可能希望将所有寄存器都保存在内存中,这样您就可以更新标题信息(打印寄存器的总数、平均值或诸如此类)。我担心这种操作不能无限扩展,您可以通过运行两次源(首先累积所有辅助信息,如平均值、最大值或最小值,然后打印内容)来实现这一点,或者将所有这些信息打印到第二页上,然后在第一页中使用"="-formula引用这些信息,这样看起来信息就在页面的顶部。
https://stackoverflow.com/questions/51768967
复制相似问题