JAVA POI导出excel(三):导出zip压缩包
创始人
2025-05-29 09:14:49
0

        相信在大部分的web项目中都会有导入导出Excel的需求,之前我也写过两篇导出Excel的文章,没看过的小伙伴可以去看哈,链接也给大家放出来了:

1、导出单个sheet

2、导出多个sheet

        但是在我们日常的工作中,需求往往没这么简单,可能需要将Excel打包成压缩包再导出等等。遇到类似的需求该怎么办呢,别慌,往下看。

一、pom引用

        pom文件中,添加以下依赖

        org.apache.poipoi5.2.2compileorg.apache.poipoi-ooxml5.2.2compile

 

二、工具类util

1.ExportExcelZip

package com.***.excel;import com.***.ServiceException;
import org.apache.poi.xssf.usermodel.XSSFWorkbook;import javax.servlet.http.HttpServletResponse;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.util.Map;
import java.util.zip.ZipEntry;
import java.util.zip.ZipOutputStream;/*** @description: 导出多个Excel 压缩成ZIP*/public class ExportExcelZip {/*** 以压缩包的方式下载excel** @param response    响应* @param workbookMap* @param zipName     压缩包名称*/public static void downloadExcelForZip(HttpServletResponse response, Map workbookMap, String zipName) throws IOException {// 文件名外的双引号处理firefox的空格截断问题ZipOutputStream out = null;try {response.setContentType("application/*");response.setHeader("content-disposition", "attachment;filename=" + new String(zipName.getBytes("gb2312"), "ISO8859-1"));response.setCharacterEncoding("UTF-8");out = new ZipOutputStream(response.getOutputStream());for (String fileName : workbookMap.keySet()) {XSSFWorkbook workbook = workbookMap.get(fileName);ZipEntry entry = new ZipEntry(fileName);out.putNextEntry(entry);// 这里讲一下,workBook.write会指定关闭数据流,如果这里直接用workbook.write(out),// 下次就会抛出out已被关闭的异常,所有用ByteArrayOutputStream来拷贝一下。ByteArrayOutputStream bos = new ByteArrayOutputStream();workbook.write(bos);bos.writeTo(out);// 关闭输入流out.closeEntry();}out.flush();} catch (IOException e) {throw new ServiceException("下载文件异常"+e.getMessage(), 9000);} finally {if (out != null) {out.close();}}}}

三、相关业务代码

1.service层

    /*** 清单导出*/Map exportSales(ExportSalesDto dto);

2.impl实现类

@Overridepublic Map exportSales(ExportSalesDto dto) {List list = new ArrayList<>();ExportSalesVo exportSalesVo = new ExportSalesVo();// 表格title(表格title需要根据自己的项目需求进行修改)List titleList = Arrays.asList("单据编号", "单据序号", "客户名称", "客户税号", "客户地址电话", "客户银行账号", "客户联系方式", "商品名称", "规格型号", "计量单位", "数量", "单价", "金额", "税率", "商品编码", "使用优惠政策", "优惠政策", "零税率标识", "自行编码", "备注", "开票人", "收款人", "复核人", "折扣金额", "折扣汇总金额", "扣除额", "成品油");exportSalesVo.setTitleList(titleList);// 填充清单数据List salesList = invoiceMapper.queryInventory(dto.getInvoiceId());for (SalesListVo vo : salesList) {vo.setInvoiceNo("3000001");vo.setInvoiceSerialNum("");vo.setCustomerMobile("");vo.setUnit("吨");// 计算货物单价:金额/数量vo.setPrice(vo.getTransportMoneyReal().divide(vo.getInvoiceNum(), 5, RoundingMode.HALF_UP));vo.setRate(new BigDecimal("0.09"));vo.setGoodsCode("");vo.setIsDiscount("");vo.setDiscount("");vo.setIsZeroRate("");vo.setCustomCode("");vo.setDrawerName("");vo.setPayeeName("");vo.setReviewName("");vo.setDiscountAmount("");vo.setDiscountTotalAmount("");vo.setDeductionAmount("");vo.setRefinedOil("");}exportSalesVo.setDataList(salesList);list.add(exportSalesVo);Map workbookMap = new HashMap<>();for (ExportSalesVo vo : list) {XSSFWorkbook wk = new XSSFWorkbook();XSSFSheet sheet = wk.createSheet("清单");XSSFRow row = sheet.createRow(0);// 添加excel titleXSSFCell cell;for (int t = 0, size = vo.getTitleList().size(); t < size; t++) {cell = row.createCell(t);cell.setCellValue(vo.getTitleList().get(t));}// 在sheet中再添加1行,存放数据// 遍历将数据写入Excel中for (int t = 0, size = vo.getDataList().size(); t < size; t++) {XSSFRow row1 = sheet.createRow(t + 1);SalesListVo salesListVo = vo.getDataList().get(t);Field[] declaredFields = salesListVo.getClass().getDeclaredFields();for (int j = 0; j < declaredFields.length; j++) {cell = row1.createCell(j);Field field = declaredFields[j];field.setAccessible(true);String value = "";try {value = field.get(salesListVo).toString();} catch (IllegalAccessException e) {e.printStackTrace();}cell.setCellValue(value);}}workbookMap.put("清单.xlsx", wk);}return workbookMap;}

3.controller层

        controller层的代码需要注意的是:

1.因为导出Excel一般都是通过浏览器进行下载的,所以入参中需要加入HttpServletResponse

2.调用封装的工具类ExportExcelZip中的downloadExcelForZip方法就可以了

    /*** 导出销货清单(压缩包)*/@GetMapping("/exportSales")public void exportSales(@Valid ExportSalesDto dto, HttpServletResponse response) {log.info("导出清单,入参:{}", dto);Map workbookMap = invoiceReviewService.exportSales(dto);try {ExportExcelZip.downloadExcelForZip(response, workbookMap, "sales.zip");} catch (IOException e) {e.printStackTrace();}}

PS:我这边只是提供了导出zip的demo,里面的表格title、数据等需要根据自己的项目需求替换成自己的代码,切记!!!

最终导出的效果:

 

 

 

 如果这篇文章对您有所帮助,或者有所启发的话,求一键三连:点赞、评论、收藏➕关注,您的支持是我坚持写作最大的动力。

相关内容

热门资讯

银行、消金公司助贷余额增速不得... 近日,中国证券报记者从多位业内人士处独家获悉,5月以来,多地金融监管部门对部分中小银行、消金公司下达...
朱鸿接任陈航,担任钉钉科技有限... 消费日报-今朝新闻讯 天眼查显示,6月23日,钉钉科技有限公司发生工商变更,陈航卸任法定代表人、董事...
3日累跌超20%,德创环保:公... 6月25日, 德创环保(603177.SH)公告,公司股票于2026年6月23日、6月24日和6月2...
北京发布2026年第七轮拟供商... 央广网北京6月25日消息(记者门庭婷)6月25日,北京市规划和自然资源委员会网站发布了2026年第七...
开放麦 | 启明创投胡奇:从A... “2026年,创投圈的浪潮再次翻涌:AI从技术概念走进产业深水区,硬科技创业从“小众赛道” 变成“主...
腾讯孙忠怀:在行业转身处 6月24日,2026腾讯视频年度发布在上海举行。腾讯公司副总裁、腾讯在线视频董事长孙忠怀以《在行业转...
加息,突变!美联储,重磅传来!... 美联储政策路径突生变数。 美国商务部经济分析局最新公布的数据显示,5月个人消费支出(PCE)物价指数...
6月合肥上门收金必看!5步避坑... 2026年6月,合肥黄金市场持续高位运行,不少市民翻出家里闲置的旧金饰、投资金条想变现,上门回收因为...
潮汕女富豪挂帅后加码液冷!祥鑫... 潮汕女强人,带着百亿公司加码液冷散热。 6月24日晚间,祥鑫科技(002965.SZ)公告称,公司董...
马斯克向太空要电,GobiX ... 一场关于「去哪里找电」的全球竞赛,正在朝两个方向展开。 作者|周永亮 编辑| 郑玄 「太空光伏是不是...
原料药行业陷入周期低谷 有药企... 每经记者|许立波 每经编辑|魏文艺 “过完年到现在,我们整个团队每个月都在出差,跑遍了亚非拉、欧美市...
家门口筛查白内障!永顺泽家镇暖... 大众卫生报·新湖南客户端6月25日讯(通讯员 彭雪姣)为切实解决辖区老年性白内障患者异地就医奔波、就...
终于等到!油价马上再大跌,这个... 点击添加图片描述(最多60个字) 编辑 各位车主朋友,好消息接二连三! 继6月18日油价大幅下调...
丈量出海新路 世界酒庄影响力指... 长期以来,全球酒庄评价体系由西方机构主导,且大多局限于单一酒种、单一评价维度,这一局面正逐渐被打破。...
峰瑞资本创始合伙人李丰:从资本... “2026年,创投圈的浪潮再次翻涌:AI从技术概念走进产业深水区,硬科技创业从“小众赛道” 变成“主...
原创 A... 迈向成熟,还有茁壮成长的机会。 作者 | 方璐 编辑丨于婞 来源 | 野马财经 2026年6月21日...
为企业解锁出海新通道!亚太中小... 6月24日下午,作为2026年APEC中小企业工商论坛的重要组成部分,亚太中小企业国际化合作发展论坛...
君赛生物港股IPO,增聘兴证国... 跟丰宜科技一样,正冲刺港股IPO的上海君赛生物股份有限公司(简称“君赛生物”)增聘一位整体协调人。 ...
圣邦股份明日上市:暗盘涨24%... 雷递网 雷建平 6月25日 圣邦微电子(北京)股份有限公司(简称:“圣邦股份”,股票代码:“0366...
科技“吃肉”,券商跟着“喝汤”... 当科技持续成为市场核心主线,押中硬科技项目的券商也成为被追逐的焦点。 6月24日,半导体零部件概念股...