将大量数据写入excel:超出了GC开销限制

时间:2018-03-02 10:11:15

标签: java apache-poi

我有一个从MongoDB读取的字符串列表(~200k行) 然后我想用Java代码将其写入excel文件:

public class OutputToExcelUtils {

    private static XSSFWorkbook workbook;
    private static final String DATA_SEPARATOR = "!";

    public static void clusterOutToExcel(List<String> data, String outputPath) {

        workbook = new XSSFWorkbook();
        FileOutputStream outputStream = null;

        writeData(data, "Data");


        try {
            outputStream = new FileOutputStream(outputPath);            
            workbook.write(outputStream);
            workbook.close();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    public static void writeData(List<String> data, String sheetName) {

        int rowNum = 0;
        XSSFSheet sheet = workbook.getSheet(sheetName);     
        sheet = workbook.createSheet(sheetName);


        for (int i = 0; i < data.size(); i++) {
            System.out.println(sheetName + " Processing line: " + i);
            int colNum = 0;
            // Split into value of cell
            String[] valuesOfLine = data.get(i).split(DATA_SEPERATOR);

            Row row = sheet.createRow(rowNum++);

            for (String valueOfCell : valuesOfLine) {
                Cell cell = row.createCell(colNum++);
                cell.setCellValue(valueOfCell);
            }
        }
    }

}

然后我收到错误:

  

线程中的异常&#34; main&#34; java.lang.OutOfMemoryError:GC开销   限制超过   org.apache.xmlbeans.impl.store.Cur $ Locations。(Cur.java:497)at at   org.apache.xmlbeans.impl.store.Locale。(Locale.java:168)at   org.apache.xmlbeans.impl.store.Locale.getLocale(Locale.java:242)at   org.apache.xmlbeans.impl.store.Locale.newInstance(Locale.java:593)at   org.apache.xmlbeans.impl.schema.SchemaTypeLoaderBase.newInstance(SchemaTypeLoaderBase.java:198)     在   org.apache.poi.POIXMLTypeLoader.newInstance(POIXMLTypeLoader.java:132)     在   org.openxmlformats.schemas.spreadsheetml.x2006.main.CTRst $ Factory.newInstance(未知   来源)at   org.apache.poi.xssf.usermodel.XSSFRichTextString。(XSSFRichTextString.java:87)     在   org.apache.poi.xssf.usermodel.XSSFCell.setCellValue(XSSFCell.java:417)     在   ups.mongo.excelutil.OutputToExcelUtils.writeData(OutputToExcelUtils.java:80)     在   ups.mongo.excelutil.OutputToExcelUtils.clusterOutToExcel(OutputToExcelUtils.java:30)     at ups.mongodb.App.main(App.java:74)

请给我一些建议吗?

谢谢你的尊重。

更新解决方案:使用SXSSWorkbook代替XSSWorkbook

public class OutputToExcelUtils {

    private static SXSSFWorkbook workbook;
    private static final String DATA_SEPERATOR = "!";

    public static void clusterOutToExcel(ClusterOutput clusterObject, ClusterOutputTrade clusterOutputTrade,
            ClusterOutputDistance ClusterOutputDistance, String outputPath) {

        workbook = new SXSSFWorkbook();
        workbook.setCompressTempFiles(true);
        FileOutputStream outputStream = null;

        writeData(clusterOutputTrade.getTrades(), "Data");

        try {
            outputStream = new FileOutputStream(outputPath);            
            workbook.write(outputStream);
            workbook.close();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    public static void writeData(List<String> data, String sheetName) {

        int rowNum = 0;
        SXSSFSheet sheet = workbook.createSheet(sheetName);
        sheet.setRandomAccessWindowSize(100); // For 100 rows saved in memory, it will flushed after wirtten to excel file

        for (int i = 0; i < data.size(); i++) {
            System.out.println(sheetName + " Processing line: " + i);
            int colNum = 0;
            // Split into value of cell
            String[] valuesOfLine = data.get(i).split(DATA_SEPERATOR);

            Row row = sheet.createRow(rowNum++);

            for (String valueOfCell : valuesOfLine) {
                Cell cell = row.createCell(colNum++);
                cell.setCellValue(valueOfCell);
            }
        }
    }

}

3 个答案:

答案 0 :(得分:3)

您的应用程序花费了太多时间进行垃圾回收。这并不一定意味着它的堆空间不足;但是,相对于执行实际工作,它在GC中花费的时间太多,因此Java运行时会将其关闭。

尝试使用以下JVM选项启用吞吐量收集

-XX:+UseParallelGC

在您熟悉它的同时,为应用程序提供尽可能多的堆空间:

-Xms????m

(其中????代表以MB为单位的堆空间量,例如-Xms8192m

如果这没有帮助,请尝试使用此选项设置更宽松的吞吐量目标:

-XX:GCTimeRatio=19 

这指定您的应用程序应该比GC相关工作多做19倍的有用工作,即它允许GC消耗最多5%的处理器时间(我认为更严格的1%默认目标可能导致上述运行时错误)

不保证他会工作。你可以检查并回复,以便其他遇到类似问题的人可以受益吗?

修改

您的根本问题仍然是您需要在构建它时将整个spreadhseet及其所有相关对象保存在内存中。另一种解决方案是序列化数据,即编写实际的电子表格文件,而不是在内存中构建它并在最后保存它。但是,这需要读取XLXS格式并创建自定义解决方案。

另一种选择是寻找内存不足的库(如果存在)。 POI的可能替代方案是 JExcelAPI (开源)和 Aspose.Cells (商业)。

我多年前使用过JExcelAPI并且有过积极的体验(但是,它似乎比POI更不积极维护,因此可能不再是最佳选择)。

编辑2

看起来POI提供了一个流媒体模型(https://poi.apache.org/spreadsheet/how-to.html#sxssf),因此这可能是最好的整体方法。

答案 1 :(得分:1)

尽量不要将所有数据加载到内存中。即使200k行的二进制表示不是那么大,内存中的隐藏对象也可能太大。正如你有一个Pojo的提示,这个pojo中的每个属性都有一个指针,每个指针取决于它是否被压缩或未被压缩将需要4或8个字节。这意味着,如果您的数据是仅具有指针的4个属性的Pojo,您将花费200 000 * 4字节(或8字节)。

理论上你可以增加JVM的内存量,但这不是一个好的解决方案,或者更准确地说它不是Live系统的好解决方案。对于非交互式系统可能没问题。

提示:使用-Xmx -Xms jvm参数来控制堆大小。

答案 2 :(得分:0)

不是从数据中获取整个列表,而是按行顺序迭代。 如果过于繁琐,请将列表写入文件,然后按行重新读取,例如Stream<String>

  Path path = Files.createTempFile(...);
  Files.write(path, list, StandarCharsets.UTF_8);
  Files.lines(path, StandarCharsets.UTF_8)
     .forEach(line -> { ... });

在Excel方面:虽然xlsx使用共享字符串,但如果XSSF不小心, 以下将对重复的字符串值使用单个String实例。

public class StringCache {
    private static final int MAX_LENGTH = 40;
    private Map<String, String> identityMap = new Map<>();

    public String cached(String s) {
         if (s == null) {
             return null;
         }
         if (s.length() > MAX_LENGTH) {
             return s;
         }
         String t = identityMap.get(s);
         if (t == null) {
             t = s;
             identityMap.put(t, t);
         }
         return t;
    }
}

StringCache strings = new StringCache();

       for (String valueOfCell : valuesOfLine) {
            Cell cell = row.createCell(colNum++);
            cell.setCellValue(strings.cached(valueOfCell));
       }