JAVA编程中用Apache POI 怎么用SXSSFWorkbook对已存在的excel(.xlsx) *** 作进行写数据 *** 作

JAVA编程中用Apache POI 怎么用SXSSFWorkbook对已存在的excel(.xlsx) *** 作进行写数据 *** 作,第1张

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解析中受益,这对大型文档的解析是个巨大优势。


欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/tougao/7899693.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-04-11
下一篇 2023-04-11

发表评论

登录后才能评论

评论列表(0条)

保存