常用开发功能——批量文件导出

news2025/1/4 18:58:16

        某天工作接到一个需求:批量导出场站的订单信息,一个场站一个Excel文档;

        与单个文件导出的区别在于,单个导出一次性只导出一个文件,在输出流中写入这一个文件即可,那么如何一次性导出多个文件?难道一次接口返回能被循环读取重复打开保存对话框?

        不是,其实是使用压缩文件,在内存中将生成的EXCEL文件依次加入到压缩文件中,最后形成一个.zip格式的文件,写入输出流作为接口的返回;

理解简单,话不多说,代码如下:

	public void exportLogBatch(ChargingParkCouponLogVO chargingParkCouponLog, HttpServletResponse response) throws IOException {
		// 设置响应头,指定下载的文件类型为压缩文件
		response.setContentType("application/zip");
		String start = DateUtil.format(chargingParkCouponLog.getStartDate(), "yyyy-MM-dd");
		String end = DateUtil.format(chargingParkCouponLog.getEndDate(), "yyyy-MM-dd");
		response.setHeader("Content-Disposition", "attachment; filename=\"订单停车券按场站批量导出(" + start + "~" + end + ").zip\"");
		String[] parkIdArr = chargingParkCouponLog.getParkIds().split(",");
		try (ZipOutputStream zipOut = new ZipOutputStream(response.getOutputStream())) {
			for (String s : parkIdArr) {
				try {
					ChargingParkConfig chargingParkConfig = chargingParkConfigService.getById(Long.valueOf(s));
					if (ObjectUtils.isEmpty(chargingParkConfig)) {
						continue;
					}
					List<ChargingParkCouponLogExport> exportList = generateExportFiles(chargingParkCouponLog);
					byte[] bytes = generateFileBytes(exportList);
					ByteArrayInputStream byteArrayInputStream = new ByteArrayInputStream(bytes);
					// 创建压缩文件的条目,并将文件内容写入压缩文件
					ZipEntry zipEntry = new ZipEntry("道闸发券识别异常清单—" + chargingParkConfig.getParkName() + ".xlsx");
					zipOut.putNextEntry(zipEntry);

					byte[] buffer = new byte[1024];
					int length;
					while ((length = byteArrayInputStream.read(buffer)) >= 0) {
						zipOut.write(buffer, 0, length);
					}
					zipOut.closeEntry();
				} catch (Exception e) {
					throw new ServiceException("停车券发放记录列表导出异常,编号:" + s);
				}
			}
			zipOut.finish();
		} catch (Exception e) {
			e.printStackTrace();
		}
	}

	private List<ChargingParkCouponLogExport> generateExportFiles(ChargingParkCouponLogVO chargingParkCouponLog) {
		Page page = new Page<>();
		page.setSize(10000L);
		List list = chargingParkCouponLogService.selectChargingParkCouponLogPage(page, chargingParkCouponLog).getRecords();
		List<ChargingParkCouponLogExport> exportList = new LinkedList<>();
		list.forEach(item -> {
			ChargingParkCouponLogExport logExport = new ChargingParkCouponLogExport();
			BeanUtils.copyProperties(item, logExport);
			exportList.add(logExport);
		});
		return exportList;
	}

	private byte[] generateFileBytes(List<ChargingParkCouponLogExport> exportList) {
		// 创建一个ByteArrayOutputStream对象,用于将Excel数据写入内存中的字节数组
		ByteArrayOutputStream outputStream = new ByteArrayOutputStream();

		// 使用ExcelWriter创建一个写入器,并将其与ByteArrayOutputStream关联
		ExcelWriter excelWriter = EasyExcel.write(outputStream, ChargingParkCouponLogExport.class).build();

		// 创建一个WriteSheet对象,设置Sheet名称和表头类
		WriteSheet writeSheet = EasyExcel.writerSheet().build();

		// 将数据列表写入Excel文件
		excelWriter.write(exportList, writeSheet);

		// 关闭ExcelWriter
		excelWriter.finish();

		// 获取ByteArrayOutputStream的字节数组
		byte[] bytes = outputStream.toByteArray();

		return bytes;
	}

调用接口:

 

返回结果:

总结下导出逻辑的变更:

         之前:准备要导出的场站ID,获取该场站要导出的数据,调用导出工具类写输出流;

         之后:准备要导出的场站的批量的ID列表,遍历ID列表循环获取场站各自要导出的数据,打开请求的总输出流作为ZipOutStream的输入流,对每个要导出的场站列表,创建单独的二进制输出流,将各自要导出的数据,写入输出流中,写入完后将输出流返回为二进制数组,作为输入流,写入创建的压缩文件中,将所有的导出文件二进制数据都写入各自的ZipEntry,最后写入总的ZipOutStream,循环执行,即可导出所有的文件作为压缩包;

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/715243.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

茶叶小程序怎么做

茶叶小程序商城是一个专为茶叶爱好者打造的购物平台&#xff0c;提供了一系列便利的功能&#xff0c;使用户能够轻松选购适合自己口味的优质茶叶。以下是该小程序商城的主要功能介绍&#xff1a; 1. 商品展示&#xff1a;在茶叶小程序商城中&#xff0c;用户可以浏览各种各样的…

点云滤波Filtering

直通滤波 就是设置一个x、y、z方向的一个取值范围&#xff0c;以过滤掉明显不在测试距离范围的点云&#xff1b;使用Intel RealSense或者激光雷达采集的数据&#xff0c;可设置一个x,y,z合理的范围&#xff0c;过滤掉我们不需要的点云。 pcl::PassThrough 是点云库&#xff08;…

4通道AD采集子卡模块有哪些推荐?

FMC134是一款4通道3.2GSPS&#xff08;2通道6.4GSPS&#xff09;采样率12位AD采集FMC子卡模块&#xff0c;该板卡为FMC标准&#xff0c;符合VITA57.4规范&#xff0c;可以作为一个理想的IO模块耦合至FPGA前端&#xff0c;16通道的JESD204B接口通过FMC连接器连接至FPGA的高速串行…

IDEA在已有项目中新建module

1.在已经新建的项目名上右击&#xff0c;选择New->Module。 2.Location选择项目地址&#xff0c;Type选择Maven&#xff0c;Language选择Java&#xff0c;Group输入com.组名&#xff08;包名&#xff09;&#xff0c;Artifact输入项目名&#xff0c;Java选择8&#xff0c;Pa…

Java中字符串与byte数组之间的相互转换

前言 java与其他语言编写的程序进行tcp/ip socket通讯时&#xff0c;通讯内容一般都转换成byte数组型&#xff0c;java在字符与数组转换也是非常方便的。下面跟我一起来了解一下字符串与byte之间转换的原理 原理 我们都知道&#xff0c;在Java里byte类型是占用1个字节&#…

【每日随笔】摩托车驾驶 ③ ( 科目三教学 | 起步 | 人行横道 | 掉头 | 停车 )

文章目录 一、科目三教学1、推车 ( 找准车辆停放位置 )2、上车前检查 ( 开始考试前准备 )3、科目三考试开始4、科目三路线介绍5、起步6、人行横道7、掉头8、停车 一、科目三教学 进去后 , ① 先起步 , ② 然后遇到人行横道 , ③ 再后掉头 , ④ 最后靠边停车 ; 1、推车 ( 找准车…

SpringSecurity_day3_授权管理

SpringSecurity和JWT SpingSecurity用于保护web安全,实现访问控制的功能,身份认证和授权操作,账号密码校验,使用token授权 JWT:可以实现跨域身份验证和授权,安全和方便 集成授权的操作流程 1.重写UserDetails中的方法getAuthorities 2.查询当前用户所有的权限信息,并返回Gr…

科技项目验收测试报告有什么注意事项和疑惑?

科技项目验收测试报告是一份重要的文件&#xff0c;用于评估科技项目的质量和可靠性&#xff0c;对项目的成功交付具有关键作用。在项目完成的最后阶段&#xff0c;通过对项目进行全面测试和评估&#xff0c;以确保项目符合预期的目标和需求&#xff0c;并满足用户的期望。 一…

π221N61 低功耗5.0kVrms 双向I²C隔离器 兼容Si8602AD-B-IS

π221N61荣湃深力科兼容IC接口的低功耗双 向隔离器&#xff0c;IC隔离器输入和输出采用二氧化硅(SiO2) 介质隔离&#xff0c;可阻断高电压并防止噪声电流进入控制侧&#xff0c;避 免电路干扰和损坏敏感器件。π221N61 是基于荣湃智能分压专利技术设计 的产品&#xff0c;与光电…

环卫工人儿子高考687分被多校争抢 父亲:就算贷款、卖房也会让他读下去

大家好&#xff01;我是老洪。 今天一早看到一则资讯。 广州一名环卫工人的儿子高考取得了687分的优异成绩&#xff0c;考入了上海交通大学。 真优秀。 据报道&#xff0c;清华大学、浙江大学、复旦大学、中科大等多所中国知名高校都曾亲自打电话向这位环卫父亲的儿子表示祝贺并…

【MySQL】不允许你不会使用子查询

&#x1f3ac; 博客主页&#xff1a;博主链接 &#x1f3a5; 本文由 M malloc 原创&#xff0c;首发于 CSDN&#x1f649; &#x1f384; 学习专栏推荐&#xff1a;LeetCode刷题集&#xff01; &#x1f3c5; 欢迎点赞 &#x1f44d; 收藏 ⭐留言 &#x1f4dd; 如有错误敬请指…

Web_php_unserialize

源码解析 依旧是反序化漏洞&#xff0c;本源码定义了一个Demo的类&#xff0c;里包含了__construct、__destruct()、__wakeup三个方法 简介&#xff1a; __construct()方法是在创建对象时&#xff0c;调用赋初值 __destruct()方法是在对象不再使用时自动调用&#xff0c;这里的…

Jest单元测试Vue项目实践

​ 做单元测试的优点&#xff1a; 1.减少bug避免低级错误 2.提高代码运行质量 3.快速定位问题 4.减少调试时间&#xff0c;提高开发效率 5.便于重构 Jest安装&#xff1a; npm install babel-jest jest jest-serializer-vue vue/test-utils vue/cli-plugin-unit-jest -D…

好用到飞起的新项目「GitHub 热点速览」

作者&#xff1a;HelloGitHub-小鱼干 虽然本周 GitHub 热榜都是一些熟悉的面孔&#xff0c;但还是有不少新开源的项目&#xff0c;比如受启发于 Stripe IDs 的 UUIDv7 扩展 typeid&#xff0c;相信有了它&#xff0c;数据标识问题就迎刃而解了。此外&#xff0c;还有刚开源就获…

Linux--自动化的构建项目:make、Makefile

make是一个命令 Makefile是一个文件 Makefile的构成&#xff1a; ①依赖关系 ②依赖方法 编写Malefile文件的最终目标是生成项目&#xff0c;换句话说就是&#xff0c;想让Makefile把我的源代码编译&#xff0c;自动形成可执行文件 示例&#xff1a; 注&#xff1a;.PHONY…

c++11 标准模板(STL)(std::basic_ostream)(五)

定义于头文件 <ostream> template< class CharT, class Traits std::char_traits<CharT> > class basic_ostream : virtual public std::basic_ios<CharT, Traits> 类模板 basic_ostream 提供字符流上的高层输出操作。受支持操作包含有格式…

C++动态库使用

个人博客地址: https://cxx001.gitee.io 前言 Windows与Linux下面的动态链接库区别 1. 文件后缀不同 Linux动态库的后缀是 .so 文件&#xff0c;而window则是 .dll 文件。 2. 文件格式不同 &#xff08;a&#xff09;Linux下是ELF格式&#xff0c;即Executable and Linkab…

数据结构--字符串的朴素模式匹配算法

数据结构–字符串的朴素模式匹配算法 主串&#xff1a; \color{purple}主串&#xff1a; 主串&#xff1a; ‘嘿嘿嘿红红火火恍恍惚惚嗨皮开森猴开森 笑出猪叫 \color{red}笑出猪叫 笑出猪叫哈哈哈哈嗨森哈哈哈哈哈哈嗝’ 模式串&#xff1a; \color{purple}模式串&#xff1a…

计算机毕业论文内容参考|基于Python的城乡低保信息管理系统的设计和实现

文章目录 导文摘要课题背景国内外现状与趋势课题内容相关技术与方法介绍系统分析系统设计系统实现系统测试总结与展望1本文总结2后续工作展望导文 计算机毕业论文内容参考|基于Python的城乡低保信息管理系统的设计和实现 摘要 本文介绍了基于Python的城乡低保信息管理系统的设…

【电路原理学习笔记】第2章:电压、电流和电阻:2.2 电荷

第2章&#xff1a;电压、电流和电阻 2.2 电荷 电子是最小的带负电荷的粒子。当物质中存在过量的电子时&#xff0c;该物质就带负的电荷&#xff1b;当电子不足时&#xff0c;就带正的净电荷。电子和质子的电荷量相等&#xff0c;但极性相反。 电荷&#xff1a;电荷是由于物质…