easyexcel/abouteasyexcel.md
jipengfei.jpf a822d53a3d update
2018-02-06 14:03:14 +08:00

4.7 KiB
Raw Blame History

easyexcel要去解决的问题

Excel读写时候内存溢出

虽然POI是目前使用最多的用来做excel解析的框架但这个框架并不那么完美。大部分使用POI都是使用他的userModel模式。userModel的好处是上手容易使用简单随便拷贝个代码跑一下剩下就是写业务转换了虽然转换也要写上百行代码相对比较好理解。然而userModel模式最大的问题是在于非常大的内存消耗一个几兆的文件解析要用掉上百兆的内存。现在很多应用采用这种模式之所以还正常在跑一定是并发不大并发上来后一定会OOM或者频繁的full gc。

其他开源框架使用复杂

对POI有过深入了解的估计才知道原来POI还有SAX模式。但SAX模式相对比较复杂excel有03和07两种版本两个版本数据存储方式截然不同sax解析方式也各不一样。想要了解清楚这两种解析方式才去写代码测试估计两天时间是需要的。再加上即使解析完要转换到自己业务模型还要很多繁琐的代码。总体下来感觉至少需要三天由于代码复杂后续维护成本巨大。

其他开源框架存在一些BUG修复不及时

由于我们的系统大多数都是大并发的情况下运行的在大并发情况下我们会发现poi存在一些bug,如果让POI团队修复估计遥遥无期了。所以我们在easyexcel对这些bug做了规避。 如下一段报错就是在大并发情况下poi抛的一个异常。

Caused by: java.io.IOException: Could not create temporary directory '/home/admin/dio2o/.default/temp/poifiles'
        at org.apache.poi.util.DefaultTempFileCreationStrategy.createTempDirectory(DefaultTempFileCreationStrategy.java:93) ~[poi-3.15.jar:3.15]
        at org.apache.poi.util.DefaultTempFileCreationStrategy.createPOIFilesDirectory(DefaultTempFileCreationStrategy.java:82) ~[poi-3.15.jar:3.15]

报错地方poi源码如下

    private void createTempDirectory(File directory) throws IOException {
        if (!(directory.exists() || directory.mkdirs()) || !directory.isDirectory()) {
            throw new IOException("Could not create temporary directory '" + directory + "'");
        }
    }

仔细看代码容易明白如果在并发情况下如果2个线程同时判断directory.exists()都 为false,但执行directory.mkdirs()如果一些线程优先执行完另外一个线程就会返回false。最终 throw new IOException("Could not create temporary directory '" + directory + "'")。针对这个问题easyexcel在写文件时候首先创建了该临时目录避免poi在并发创建时候引起不该有的报错。

Excel格式分析格式分析

  • xls是Microsoft Excel2007前excel的文件存储格式实现原理是基于微软的ole db是微软com组件的一种实现本质上也是一个微型数据库由于微软的东西很多不开源另外也已经被淘汰了解它的细节意义不大底层的编程都是基于微软的com组件去开发的。
  • xlsx是Microsoft Excel2007后excel的文件存储格式实现是基于openXml和zip技术。这种存储简单安全传输方便同时处理数据也变的简单。
  • csv 我们可以理解为纯文本文件可以被excel打开。他的格式非常简单解析起来和解析文本文件一样。

核心原理

写有大量数据的xlsx文件时POI为我们提供了SXSSFWorkBook类来处理这个类的处理机制是当内存中的数据条数达到一个极限数量的时候就flush这部分数据再依次处理余下的数据这个在大多数场景能够满足需求。 读有大量数据的文件时使用WorkBook处理就不行了因为POI对文件是先将文件中的cell读入内存生成一个树的结构针对Excel中的每个sheet使用TreeMap存储sheet中的行。如果数据量比较大则同样会产生java.lang.OutOfMemoryError: Java heap space错误。POI官方推荐使用“XSSF and SAXevent API”方式来解决。 分析清楚POI后要解决OOM有3个关键。

1、文件解压文件读取通过文件形式

屏幕快照 2018-01-22 上午8.52.08.png

2、避免将全部全部数据一次加载到内存

采用sax模式一行一行解析并将一行的解析结果以观察者的模式通知处理。 基础模板1 (2).png

3、抛弃不重要的数据

Excel解析时候会包含样式字体宽度等数据但这些数据是我们不关系的如果将这部分数据抛弃可以大大降低内存使用。Excel中数据如下Style占了相当大的空间。