XSSFWorkbook wb=new XSSFWorkbook(参数)中的参数是InputStream ,你直接XSSFWorkbook wb=new XSSFWorkbook(fs)就可以了。
第一步查询数据--这一步读者自行实现自己的数据查询 List<PointInfo>points = null
points = this.dao.getAllCollect(userId)
final Map<String, List<PointInfo>>pointMap = new HashMap<>()
for (final PointInfo pointInfo : points) {
final String pt = pointInfo.getPointType() if (pointMap.containsKey(pt)) {final List<PointInfo>subList = pointMap.get(pt)
subList.add(pointInfo)
} else {final List<PointInfo>subList = new ArrayList<>()subList.add(pointInfo)
pointMap.put(pt, subList
第二步:生成工作簿
final SXSSFWorkbook wb = new SXSSFWorkbook()
// 对每一种类型生成一个sheet
for (final Map.Entry<String, List<PointInfo>>entry : pointMap.entrySet()) {
final List<PointInfo>pts = entry.getValue()
// 获取每种类型的名字--作为sheet显示名称--如果不需要分sheet可忽略
String typeName = ""
if (this.dao.getTypeByTypeCode(pts.get(0).getPointType()) != null) {
typeName = this.dao.getTypeByTypeCode(pts.get(0).getPointType()).getPointTypeName()
}
final Sheet sheet = wb.createSheet(typeName)
//生成用于插入图片的容器--这个方法返回的类型在老api中不同
final Drawing patriarch = sheet.createDrawingPatriarch()
// 为sheet1生成第一行,用于放表头信息
final Row row = sheet.createRow(0)
// 第一行的第一个单元格的值
Cell cell = row.createCell((short) 0)
cell.setCellValue("详细地址")
cell = row.createCell((short) 1)
cell.setCellValue("经度")
cell = row.createCell((short) 2)
cell.setCellValue("纬度")
cell = row.createCell((short) 3)
for (int i = 0i <pts.size()i++) {
final Row each = sheet.createRow(i + 1)
Cell infoCell = each.createCell((short) 0)
infoCell.setCellValue(pts.get(i).getAddrDetail())
infoCell = each.createCell((short) 1)
infoCell.setCellValue(pts.get(i).getX())
infoCell = each.createCell((short) 2)
infoCell.setCellValue(pts.get(i).getY())
infoCell = each.createCell((short) 3)
//查询获取图片路径信息--该步读者自定义
PointPic pic = this.dao.getPicInfoByPointId(pts.get(i).getId())
try {
if (pic != null) {
for (int k = 0k <6k++) {//因为有六张图片,所以循环6次
final short colNum = (short) (4+k)
infoCell = each.createCell(colNum)
BufferedImage img = null
switch (k) {
case 0:
if (!StringUtils.isEmpty(pic.getPicOneAddr())) {
File imgFile = new File(pic.getPicOneAddr())
img = ImageIO.read(imgFile)
imgFile = null
}
break
case 1:
if (!StringUtils.isEmpty(pic.getPicTwoAddr())) {
File imgFile = new File(pic.getPicTwoAddr())
img = ImageIO.read(imgFile)
imgFile = null
}
break
case 2:
if (!StringUtils.isEmpty(pic.getPicThreeAddr())) {
File imgFile = new File(pic.getPicThreeAddr())
img = ImageIO.read(imgFile)
imgFile = null
}
break
case 3:
if (!StringUtils.isEmpty(pic.getPicFourAddr())) {
File imgFile = new File(pic.getPicFourAddr())
img = ImageIO.read(imgFile)
imgFile = null
}
break
case 4:
if (!StringUtils.isEmpty(pic.getPicFiveAddr())) {
File imgFile = new File(pic.getPicFiveAddr())
img = ImageIO.read(imgFile)
imgFile = null
}
break
case 5:
if (!StringUtils.isEmpty(pic.getPicSixAddr())) {
File imgFile = new File(pic.getPicSixAddr())
img = ImageIO.read(imgFile)
imgFile = null
}
break
}
ByteArrayOutputStream byteArrayOut = new ByteArrayOutputStream()
ImageIO.write(img, "jpg", byteArrayOut)
img = null
//设置每张图片插入位置
final XSSFClientAnchor anchor = new XSSFClientAnchor(0, 0, 0, 0, colNum,
i + 1, (short) (colNum + 1), i + 2)//参数为图片插入在表格的坐标,可以自行查看api研究参数
anchor.setAnchorType(0)
// 插入图片
patriarch.createPicture(anchor, wb.addPicture(
byteArrayOut.toByteArray(), HSSFWorkbook.PICTURE_TYPE_JPEG))
byteArrayOut.close()
byteArrayOut = null
}
pic = null
}
} catch (final Exception e) {
e.printStackTrace()
}
}
}
final ByteArrayOutputStream os = new ByteArrayOutputStream()
try {
wb.write(os)
} catch (final IOException e) {
e.printStackTrace()
}
final byte[] content = os.toByteArray()
final String url = Var.BASE_URL+ File.separator + "output.xls"//读者自定义路径
final File file = new File(url)// Excel文件生成后存储的位置。
OutputStream fos = null
try {
fos = new FileOutputStream(file)
fos.write(content)
os.close()
fos.close()
} catch (final Exception e) {
e.printStackTrace()
}
return url//文件保存成功
1.创建工作簿Workbook
2.创建Sheet
3.创建行Row
4.创建单元格Cell
HSSFWorkbook是 *** 作Excel2003以前(包括2003)的版本,扩展名为.xls,所以每个Sheet局限就是导出的行数至多为65535行,一般不会发生内存不足的情况(OOM)。
这种形式的出现是由于HSSFWorkbook的局限性而产生的,因为其所导出的行数比较少,并且只针对Excel2003以前(包括2003)的版本的版本,所以 XSSFWookbook应运而生,其对应的是EXCEL2007以后的版本(1048576行,16384列)扩展名.xlsx,每个Sheet最多可以导出104万行,不过这样就伴随着一个OOM内存溢出的问题,原因是你所创建的sheet row cell 等此时是存在内存中的,随着数据量增大 ,内存的需求量也就增大,那么很大可能就是要OOM了。
从POI 3.8版本开始,提供了一种基于XSSFWorkbook的低内存占用的工作簿SXSSFWorkbook。
引用官方的介绍,简单概括就是:
SXSSF是对XSSF的一种流式扩展, 特点是采用了滑动窗口的机制,低内存占用 ,主要用于数据量非常大的电子表格而虚拟机堆有限的情况。
原理是利用了滑动窗口机制。
SXSSFWorkbook.DEFAULT_WINDOW_SIZE默认值是100,表示在内存中最多存在100个Row对象,当写第101个Row对象的时候就会把第1个Row对象以XML格式写入C:\Users\wange\AppData\Local\Temp路径下的临时文件中,后面的以此类推,始终保持内存中最多存在100个Row对象。
SXSSFWorkbook默认使用内联字符串而不是 共享字符串表 (SharedStringsTable)。启用共享字符串时,文档中的所有唯一字符串都必须保存在内存中,因此会占用更多的内存。
与XSSF的对比,在一个时间点上,只可以访问一定数量的Row;不再支持Sheet.clone();不再支持公式的求值。但是除了滑动窗口,其余的EXCLE *** 作仍然使用的是XSSF的API。
另外官方提示导出EXCEL后应该调用wb.dispose()来删除之前保存的临时文件。
wb.write(out)通过源码了解到过程是
1、将wb的所有sheet调用flushRows()移出内存,写入临时.xml文件中
2、生成了一个临时.xlsx文件将wb的一些模板数据写入这个临时文件
3、将这个临时.xlsx文件转成ZipFile,遍历所有ZipEntry来获取Sheet,如果没有Sheet则直接复制流。
4、如果能够获取到Sheet的则是那些临时.xml文件,在对这些文件进行解析并追踪写入导出文件中。
(这边可能是涉及到了一些EXCEL文件格式的原理,就不深入研究了)
SXSSFWorkbook wb = new SXSSFWorkbook(-1)
初始化设置为-1的时候我们可以自己定义写临时文件规则,比如每读1000行记录flush到临时一次,可以大大减少磁盘IO次数。
使用SAX模型来解析EXCEL不像DOM模型一下把所有文件内容加载进内存,它逐行扫描文档,一边扫描,一边解析。所以那些只需要单遍读取内容的应用程序就可以从SAX解析中受益,这对大型文档的解析是个巨大优势。
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)