Easy Excel从入门到精通!!!

目录

1.文件导入

1.1基本方式读取excel文件内容

1.2注解+模型映射器读取excel

1.3多行表头读取

1.4文件上传读取

2.文件导出

2.1基本方式导出

2.2模型映射导出

2.3设置行高、列宽等内容

2.4合并单元格

2.5导出设置超链接、批注、公式

2.6模板填充对象导出

2.7模板填充对象列表导出

2.8模板组合填充

3.文件下载

EasyExcel是一个基于Java的、快速、简洁、解决大文件内存溢出的Excel处理工具。
他能让你在不用考虑性能、内存的等因素的情况下,快速完成Excel的读、写等功能

1.文件导入

1.1基本方式读取excel文件内容

read为读取,sheet就是指定excel文件中第几个sheet页,doReadnSync为结尾方法

//指定位置读取excel内容@Testpublic void test01(){File file = new File("C:\\Users\\Lenovo\\Desktop\\excel\\user-list.xlsx");List<Object> objects = EasyExcel.read(file).sheet(0).doReadSync();for (Object object : objects) {log.info("读取内容为{}",object);}}//输入流读取@Testpublic void test02(){InputStream inputStream = EazyExcelTest.class.getClassLoader().getResourceAsStream("user-list.xlsx");List<Map<Integer, Object>> list = EasyExcel.read(inputStream).sheet(0).doReadSync();for (Map<Integer, Object> item : list) {log.info("昵称: {}, 性别: {}, 生日: {}, 邮箱: {}, 积分: {}", item.get(0), item.get(1), item.get(2), item.get(3), item.get(4));}}

1.2注解+模型映射器读取excel

在上面中读取的excel返回的其实是个Map其中key为下标value为单元格数据,但是如果通过map.get()方式进行获取就太麻烦了,所以我们可以定义一个类让他去和excel文件数据进行映射这样在去操作对象就方便多了

表格为

实体类为

其中的注解起到的作用就是进行与excel表格字段进行映射下面我一一来说

@ExcelProperty(value = " "),这个注解是最基本的通过它与表格字段进行映射

@DateTimeFormat(value = "年月日时分秒"),这个注解负责和表格的日期字段进行映射

@NumberFormat(value = " ")这个注解负责数字转换,用String去接收excel数字格式的数据会调用这个注解,如果想什么结尾就在结尾加上例如%

@Data
public class UserInfoModel {/*** 昵称。*/@ExcelProperty(value = "昵称")private String userName;/*** 性别。*/@ExcelProperty(value = "性别",converter = UserInfoGenderConverter.class)private Integer userGender;/*** 生日。*/@ExcelProperty(value = "生日")@DateTimeFormat(value = "yyyy-MM-dd")private String userBirth;/*** 邮箱。*/@ExcelProperty(value = "邮箱")private String userEmail;/*** 积分。*/@ExcelProperty(value = "积分")private Integer userScore;/*** 排名。*/@ExcelProperty(value = "排名")@NumberFormat(value = "#.##%")private String userRank;
}

我们看到在注解@ExcelProperty(value =" ") 后面有个convert他的作用就是因为字段男女是单一的在excel中,我们想在程序中用数字表示就可以自定义一个模型映射器进行转换,代码如下,实现converter指定类型然后做判断返回

/*** 用户信息,性别,转换器。*/
public class UserInfoGenderConverter implements Converter<Integer> {public Integer convertToJavaData(ReadCellData<?> cellData, ExcelContentProperty contentProperty, GlobalConfiguration globalConfiguration) throws Exception {String valuee = cellData.getStringValue();switch (valuee) {case "男":return 1;case "女":return 2;default:return 0;}}
}

读取代码

  @Testpublic void test3() {File file = new File("C:\\Users\\Lenovo\\Desktop\\excel\\user-list.xlsx");List<UserInfoModel> list = EasyExcel.read(file).sheet(0).head(UserInfoModel.class).doReadSync();for (UserInfoModel item : list) {log.info("昵称: {}, 性别: {}, 生日: {}, 邮箱: {}, 积分: {},排名{}", item.getUserName(), item.getUserGender(), item.getUserBirth(), item.getUserEmail(), item.getUserScore(),item.getUserRank());}}

1.3多行表头读取

开发中有可能用户上传的表格是从第几行之后才是正式的数据,而我们在代码中默认的就是从第一行进行读取,遇到这种情况我们可以指定从第几行读取,通过headRowNumber进行实现

实例

代码

@Testpublic void test4() {File file = new File("C:\\Users\\Lenovo\\Desktop\\excel\\user-list2.xlsx");List<UserInfoModel> list = EasyExcel.read(file).headRowNumber(6).sheet(0).head(UserInfoModel.class).doReadSync();for (UserInfoModel item : list) {log.info("昵称: {}, 性别: {}, 生日: {}, 邮箱: {}, 积分: {},排名{}", item.getUserName(), item.getUserGender(), item.getUserBirth(), item.getUserEmail(), item.getUserScore(), item.getUserRank());}}

1.4文件上传读取

通过ApiPost上传文件,后台进行读取处理

@RestController
@RequestMapping("/home")
public class HomeController {@PostMapping("/upload")@SneakyThrowspublic List<UserInfoModel> upload(@RequestPart("file") MultipartFile file) {InputStream inputStream = file.getInputStream();List<UserInfoModel> list = EasyExcel.read(inputStream).head(UserInfoModel.class).headRowNumber(6).sheet(0).doReadSync();return list;}
}

2.文件导出

2.1基本方式导出

首先我们需要新建一张表用于导出时指定位置,下面演示导出三种数据方式;write指定要导出位置的excel表,sheet指定sheet名称dowrite指定导出的数据

 /*** 导出路径。*/private final String EXPORT_PATH = "C:\\Users\\Lenovo\\Desktop\\excel\\eazyExcel.xlsx";/*** 演示:List<List<Object>> 方式导出。*/@Testpublic void test01(){List<Object> list1 = Arrays.asList("郭德纲",28,"男");List<Object> list2 = Arrays.asList("郭麒麟",30,"男");List<List<Object>> asList = Arrays.asList(list1, list2);EasyExcel.write(EXPORT_PATH).sheet("export").doWrite(asList);}/*** Map导出*/@Testpublic void test02(){Map<Integer,Object> map1 = new HashMap<>();map1.put(0,"郭德纲");map1.put(1,28);map1.put(2,"男");Map<Integer,Object> map2 = new HashMap<>();map2.put(0,"郭麒麟");map2.put(1,60);map2.put(2,"男");List<Map<Integer, Object>> asList = Arrays.asList(map1, map2);EasyExcel.write(EXPORT_PATH).sheet("export1").doWrite(asList);}/*** 对象导出*/@Test@SneakyThrowspublic void test03(){UserInfoModel userInfoModel1 = new UserInfoModel();userInfoModel1.setUserBirth(DateUtils.parseDate("1997-06-29"));userInfoModel1.setUserName("郭德纲");userInfoModel1.setUserGender(28);UserInfoModel userInfoModel2 = new UserInfoModel();userInfoModel2.setUserBirth(DateUtils.parseDate("1997-06-29"));userInfoModel2.setUserName("郭麒麟");userInfoModel2.setUserGender(28);OrderInfoModel orderInfoModel = new OrderInfoModel();orderInfoModel.setOrderTime(DateUtils.parseDate("1987-06-28"));orderInfoModel.setOrderTitle("订单标题");orderInfoModel.setOrderPrice(new BigDecimal("10"));List<Object> list = Arrays.asList(userInfoModel1, userInfoModel2,orderInfoModel);EasyExcel.write(EXPORT_PATH).sheet("export2").doWrite(list);}

2.2模型映射导出

在开发中我们通常是根据前端传过来的参数去DB中查询出对应数据封装成对象集合的形式然后导出

模拟获取数据

 /*** 获取用户信息列表。** @return 返回结果。*/@SneakyThrowsprivate List<UserInfoModel> getList() {UserInfoModel user1 = new UserInfoModel();user1.setUserName("郭德纲"); // 用户姓名。user1.setUserGender(1); // 用户性别。user1.setUserBirth(DateUtils.parseDate("1973-01-18")); // 用户生日。user1.setUserScore(100); // 用户积分。user1.setUserReward(BigDecimal.valueOf(123.45)); // 用户佣金。UserInfoModel user2 = new UserInfoModel();user2.setUserName("于谦"); // 用户姓名。user2.setUserGender(2); // 用户性别。user2.setUserBirth(DateUtils.parseDate("1967-12-06")); // 用户生日。user2.setUserScore(200); // 用户积分。user2.setUserReward(BigDecimal.valueOf(234.56)); // 用户佣金。UserInfoModel user3 = new UserInfoModel();user3.setUserName("岳云鹏"); // 用户姓名。user3.setUserGender(0); // 用户性别。user3.setUserBirth(DateUtils.parseDate("1985-09-17")); // 用户生日。user3.setUserScore(300); // 用户积分。user3.setUserReward(BigDecimal.valueOf(345.67)); // 用户佣金。return Arrays.asList(user1, user2, user3);}

实体类

在导入时我们会根据excel文件的表头映射成实体类一一对应然后将list集合进行数据操作,同理导出也是一样的我们查询出的数据映射到实体类然后设置表头这样就能导出excel文件了,这里的注解和上面导入一样,其中我们可以在@ExcelProperty注解中通过{ }将字段汇总到一个表头下,@ExclIgnore注解的作用就是排除掉这个字段不让他在表格中出现

/*** 用户信息,模型。*/
@Data
public class UserInfoModel implements Serializable {private static final long serialVersionUID = 1L;/*** 用户姓名。*/@ExcelProperty(value = {"基本信息","用户姓名"})private String userName;/*** 用户性别。*/@ExcelProperty(value = {"基本信息","用户性别"},converter = UserInfoModelConvert.class)private Integer userGender;/*** 用户生日。*/@ExcelProperty(value = {"基本信息","用户生日"})@DateTimeFormat(value = "yyyy年MM月dd日")private Date userBirth;/*** 用户积分。*/@ExcelProperty(value = "用户积分")private Integer userScore;/*** 用户佣金。*/@ExcelProperty(value = "用户佣金")@NumberFormat(value = "$#.##")@ExcelIgnoreprivate BigDecimal userReward;}

导出代码

导出excludeColumnFieldNames的意思是排除调那些字段,includeColumnFieldNames是只要那些字段

  public static final String EXPORT_FILE = "C:\\Users\\Lenovo\\Desktop\\excel\\eazyExcel.xlsx";/*** 测试:模型映射导出。*/@Test@SneakyThrowspublic void testExport() {// 获取:用户信息列表。List<UserInfoModel> list = this.getList();// 处理:导出数据。EasyExcel.write(EXPORT_FILE).head(UserInfoModel.class).sheet("sheet01").excludeColumnFieldNames(Arrays.asList("userScore")).doWrite(list);}

2.3设置行高、列宽等内容

@ContentRowHeight(value = 20) //行高(内容)只能加在类上
@HeadRowHeight(value = 20) //行高 (标题)只能加在类上
@ColumnWidth(value = 20) //列宽 可以加在类上也可以加在字段上

导出代码

其中.registerWriteHandler(new LongestMatchColumnWidthStyleStrategy()) //自适应表格格式

 /*** 测试:模型映射导出。*/@Test@SneakyThrowspublic void testExport() {// 获取:用户信息列表。List<UserInfoModel> list = this.getList();// 处理:导出数据。EasyExcel.write(EXPORT_FILE).sheet("导出数据").head(UserInfoModel.class).registerWriteHandler(new LongestMatchColumnWidthStyleStrategy()) //自适应格式.doWrite(list);}

实体类

/*** 用户信息,模型。*/
@Data
@ContentRowHeight(value = 20) //行高(内容)
@HeadRowHeight(value = 20) //行高 (标题)
public class UserInfoModel implements Serializable {private static final long serialVersionUID = 1L;/*** 用户姓名。*/@ExcelProperty(value = {"基本信息", "用户姓名"})private String userName;/*** 用户性别。*/@ExcelProperty(value = {"基本信息", "用户性别"})private Integer userGender;/*** 用户生日。*/@ExcelProperty(value = {"基本信息", "用户生日"})@DateTimeFormat(value = "yyyy年MM月dd日")@ColumnWidth(value = 20)private Date userBirth;/*** 用户积分。*/@ExcelProperty(value = {"账户信息", "用户积分"})private Integer userScore;/*** 用户佣金。*/@ExcelProperty(value = {"账户信息", "用户佣金"})@NumberFormat(value = "¥#.##")private BigDecimal userReward;}

2.4合并单元格

合并单元格也很常见,共有两种方式

方式1:注解合并

@OnceAbsoluteMerge加在类上指定某个单元格进行合并

@ContentLoopMerge加在某个表头下指定数量进行循环合并

/*** 用户信息,模型。*/
@Data
//@OnceAbsoluteMerge(firstRowIndex = 2, lastRowIndex = 3, firstColumnIndex = 0, lastColumnIndex = 0) //单次指定某个单元格合并
//firstRowIndex = 起始位置为0 2的话也就是第三行的位置, lastRowIndex = 3其实位置为0就是firstRowIndex到3的位置进行合并, firstColumnIndex = 0第几列, lastColumnIndex = 0第几列结束
public class UserInfoModel implements Serializable {private static final long serialVersionUID = 1L;/*** 团队名称。*/@ExcelProperty(value = "团队名称")@ContentLoopMerge(eachRow = 2) //循环合并就是自动合并,在团队名称单元格下面自动进行每两个单元格进行合并,但是不智能可能会多出来一行private String teamName;

方式2:代码合并

.registerWriteHandler(new OnceAbsoluteMergeStrategy(0,1,4,4))//指定单元格进行合并一次
.registerWriteHandler(new LoopMergeStrategy(2,0))//循环合并
@Test@SneakyThrowspublic void test() {EasyExcel.write(EXPORT_FILE).head(UserInfoModel.class).registerWriteHandler(new LongestMatchColumnWidthStyleStrategy()).registerWriteHandler(new OnceAbsoluteMergeStrategy(0,1,4,4))//指定单元格进行合并一次.registerWriteHandler(new LoopMergeStrategy(2,0))//循环合并.sheet("sheet1").doWrite(this::getList);}

2.5导出设置超链接、批注、公式

一句话概括就是在你要设置的字段上加上原始数据类型WriteCellData<类型>,然后在代码中进行添加就可以了,如下所示;导出公式不常用这里不做演示

模型类

/*** 用户信息,模型类。*/
@Data
@ColumnWidth(value = 20)
public class UserInfoModel implements Serializable {private static final long serialVersionUID = 1L;/*** 行号。*/@ExcelIgnore@ExcelProperty(value = "行号")private Integer rowNumber;/*** 用户标识。*/@ExcelProperty(value = "用户标识")private WriteCellData<Long> id;/*** 用户昵称。*/@ExcelProperty(value = "用户昵称")private WriteCellData<String> userNickName;/*** 用户性别。*/@ExcelProperty(value = "用户性别")private Integer userGender;/*** 用户生日。*/@ExcelProperty(value = "用户生日")private Date userBirth;/*** 用户年龄。*/@ExcelIgnore@ExcelProperty(value = "用户年龄")private Integer userAge;}

导出代码

  @Testpublic void test() {List<UserInfoEntity> list = userInfoService.list();List<UserInfoModel> modelList = list.stream().map(item -> {UserInfoModel userInfoModel = new UserInfoModel();//添加批注CommentData commentData = new CommentData();commentData.setAuthor("我是作者李佳伟");commentData.setRichTextStringData(new RichTextStringData("这是批注内容"));WriteCellData<Long> id = new WriteCellData<>(new BigDecimal(item.getId()));id.setCommentData(commentData);userInfoModel.setId(id);//添加超链接HyperlinkData hyperlinkData = new HyperlinkData();hyperlinkData.setHyperlinkType(HyperlinkData.HyperlinkType.URL);hyperlinkData.setAddress("http://www.baidu.com");WriteCellData<String> userNickName = new WriteCellData<>(item.getUserNickname());userNickName.setHyperlinkData(hyperlinkData);userInfoModel.setUserNickName(userNickName);userInfoModel.setUserGender(item.getUserGender());userInfoModel.setUserBirth(item.getUserBirth());return userInfoModel;}).collect(Collectors.toList());EasyExcel.write(EXPORT_FILE).head(UserInfoModel.class).sheet().registerWriteHandler(new LongestMatchColumnWidthStyleStrategy()).doWrite(modelList);}

2.6导出图片到Excel

导出图片到Excel的需求很常见,有多种方法下面我将一一演示

模型类

如果你想通过获取图片的file、byte、流、Sting的方式获取图片然后导出到excel就分别指定他们的类型,但是这种方式不常见,因为图片一般我们都是存储在OSS中然后我们通过图片的URL来进行访问图片,所以我们指定图片为URL类型,还有一种情况存储在数据库的图片URL为Base64格式的,那么我们需要进行转换

/*** 用户信息,模型。*/
@Data
@ContentRowHeight(value = 50)
@ColumnWidth(value = 30)
public class UserInfoModel implements Serializable {private static final long serialVersionUID = 1L;@ExcelProperty(value = "File类型")private File fileImage;@ExcelProperty(value = "byte[]类型")private byte[] byteImage;@ExcelProperty(value = "InputStream类型")private InputStream inputStreamImage;@ExcelProperty(value = "URL类型")private URL urlImage;@ExcelProperty(value = "String类型", converter = Base64Convert1.class)private String stringImage;}

导出代码

如果图片通过File、byte、流则直接指定位置获取即可,如果是URL这种最常见则newURL进行指定导出、如果是通过String则需要通过convert转换成字节其实也是通过File获取然后转换这样,如果数据库图片的URL是Base64的那么也很简单我们只需自定义一个Convert,将Base64转成字节在将字节通过ConvertString类型进行返回就可以了;或者存到DB的是Base64编码后的URL然后我们可以查询出来转成URL在进行导出图片到excel

public static final String EXPORT_FILE = "C:\\Users\\Lenovo\\Desktop\\excel\\eazyExcel.xlsx";@Test@SneakyThrowspublic void test() {UserInfoModel userInfoModel = new UserInfoModel();//文件类型userInfoModel.setFileImage(new File("C:\\Users\\Lenovo\\Desktop\\excel\\20201124032511.png"));//byte类型userInfoModel.setByteImage(FileUtils.readFileToByteArray(new File("C:\\Users\\Lenovo\\Desktop\\excel\\20201124032511.png")));//输入流类型userInfoModel.setInputStreamImage(FileUtils.openInputStream(new File("C:\\Users\\Lenovo\\Desktop\\excel\\20201124032511.png")));//URL类型userInfoModel.setUrlImage(new URL("https://img-home.csdnimg.cn/images/20201124032511.png"));//String类型userInfoModel.setStringImage("iVBORw0KGgoAAAANSUhEUgAAAKAAAABYCAYAAAByDvxZAAAAAXNSR0IArs4c6QAAAERlWElmTU0AKgAAAAgAAYdpAAQAAAABAAAAGgAAAAAAA6ABAAMAAAABAAEAAKACAAQAAAABAAAAoKADAAQAAAABAAAAWAAAAADfqAIVAAAeZklEQVR4Ae1cB3hUx7W");EasyExcel.write(EXPORT_FILE).head(UserInfoModel.class).sheet().registerWriteHandler(new LongestMatchColumnWidthStyleStrategy()).doWrite(Collections.singletonList(userInfoModel));}@SneakyThrows@Testpublic void testBase64(){byte[] bytes = FileUtils.readFileToByteArray(new File("C:\\Users\\Lenovo\\Desktop\\excel\\20201124032511.png"));String base64 = Base64.getEncoder().encodeToString(bytes);System.out.println(base64);}

2.7模板填充对象导出

我们上面学习到了通过模型类指定单元格进行导出,然后指定他的宽高、格式等等,那么这种方式大家发现是不是很麻烦,还需要去指定格式等等;现在我们可以在resouce目录下创建一个模板文件然后我们在模板中设置好格式,然后将数据填充到模板中这样就方便多了,这种方式适合简单的导出没什么要求,但是如果是出现一些字段上的需求比如超链接、批注、金钱符号等就无法满足。

模板样式

模型类

/*** 用户信息,模型。*/
@Data
public class UserInfoModel implements Serializable {private static final long serialVersionUID = 1L;/*** 用户编号。*/private Integer userCode;/*** 用户昵称。*/private String userNickname;/*** 用户积分。*/private Integer userScore;/*** 用户佣金。*/private BigDecimal userReward;}

导出代码

    /*** 对象方式*/@Testpublic void test1(){InputStream inputStream = EasyExcelTest.class.getClassLoader().getResourceAsStream("export-template.xlsx");UserInfoModel userInfoModel = new UserInfoModel();userInfoModel.setUserCode(1001);userInfoModel.setUserNickname("李佳伟");userInfoModel.setUserScore(200);userInfoModel.setUserReward(BigDecimal.valueOf(123.45));EasyExcel.write(EXPORT_FILE).withTemplate(inputStream).sheet().doFill(userInfoModel);}

导出样式

2.8模板填充对象列表导出

上面我们是添加一个对象然后导出,但是实际业务是多条数据的所以我们要实现添加列表的方式,与单个对象导出不同点就是模板中多加了个.

2.9模板组合填充

在开发中我们可能会遇到往一个表格中添加数据但是表格中有多个表的情况如图所示,然后我们还可能遇到横向填充的情况,这里为什么加个前缀dept或者user呢是为了在填充模型数据时好做区分

我们先获取excelWriter获取操作对象,然后获取添加的表writeSheet,如果是填充一个对象则直接fill如果是多个对象则需要new FillWrapper然后指定你填填充的前缀加数据,然后添加表writerSheet如果需要横向填充的话那就需要手动设置一个进行填充

FillConfig config = FillConfig.builder().direction(WriteDirectionEnum.HORIZONTAL).build(); //横向填充
  @Testpublic void test() {InputStream exportTemplate = EasyExcelTest.class.getClassLoader().getResourceAsStream("export-template.xlsx");try (ExcelWriter excelWriter = EasyExcel.write(EXPORT_FILE).withTemplate(exportTemplate).build()) {WriteSheet writeSheet = EasyExcel.writerSheet().build();// 公司信息。Map<String, Object> companyInfo = new HashMap<>();companyInfo.put("companyName", "北京XXX信息技术有限公司");excelWriter.fill(companyInfo, writeSheet);Map<String, Object> deptMap1 = new HashMap<>();deptMap1.put("deptName", "研发部");deptMap1.put("deptMaster","李佳伟");deptMap1.put("deptContact","17045454589");Map<String, Object> deptMap2 = new HashMap<>();deptMap2.put("deptName", "财务部");deptMap2.put("deptMaster","刘晓宇");deptMap2.put("deptContact","17045454589");FillConfig config = FillConfig.builder().direction(WriteDirectionEnum.HORIZONTAL).build(); //横向填充List<Map<String, Object>> asList = Arrays.asList(deptMap1, deptMap2);excelWriter.fill(new FillWrapper("dept",asList), config, writeSheet);Map<String, Object> userInfo1 = new HashMap<>();userInfo1.put("userCode", 1001);userInfo1.put("userNickname", "张三");userInfo1.put("userScore", 100);userInfo1.put("userReward", BigDecimal.valueOf(123.45));Map<String, Object> userInfo2 = new HashMap<>();userInfo2.put("userCode", 1002);userInfo2.put("userNickname", "李四");userInfo2.put("userScore", 100);userInfo2.put("userReward", BigDecimal.valueOf(123.45));List<Map<String, Object>> mapList = Arrays.asList(userInfo1, userInfo2);excelWriter.fill(new FillWrapper("user",mapList),writeSheet);}}

3.文件下载

文件下载其实我们在导出的时候通常会返回给前端excel的URL下载地址,然后前端直接根据这个URL进行下载就好了我们不用去管他,但是有时候有一些需求需要单独弄一个下载按钮,而不是直接导出的时候就下载了那么也就是如果单独弄下载按钮我们导出就不用返回URL了,直接提示用户稍后去下载中心查看下载就好了。

这里提供两种下载方式,一个是不提供URL进行下载就是可以通过前端传的数据然后去数据库查询出对应数据然后下载,那么如果这种方式也就相当于导出和下载结合了然后写到response里面,另外一种是通过URL进行下载这个URL就是存储在OSS网络的表格URL前端直接通过URL进行下载

未提供URL下载方式相当于导出下载结合

   @GetMapping(value = "/export")@SneakyThrowspublic void export(HttpServletResponse response) {List<UserInfoEntity> list = this.userInfoService.list();response.setCharacterEncoding("UTF-8");response.setContentType("application/vnd.ms-excel");String fileName = URLEncoder.encode("用户信息", "UTF-8");response.setHeader("Content-disposition", "attachment;filename=" + fileName + ".xlsx");ServletOutputStream outputStream = response.getOutputStream();InputStream exportTemplate = Application.class.getClassLoader().getResourceAsStream("export-template.xlsx");EasyExcel.write(outputStream).withTemplate(exportTemplate).sheet().doFill(list);}

提供URL由前端直接去下载的方式

/*** form表单提交,下载文件流** @param response   HttpServletResponse* @param exportName 文件名* @param exportUrl  下载文件URL*/@RequestMapping(value = "/download", method = RequestMethod.POST)@ResponseBody@PermissionCheck(skipAop = true)public void download(HttpServletResponse response,@RequestParam("exportName") String exportName,@RequestParam("exportUrl") String exportUrl) {// 设置强制下载不打开response.setContentType("application/force-download");response.setHeader("Content-Disposition", "attachment; filename=" + exportName);int byteRead;try {URL url = new URL(exportUrl);URLConnection conn = url.openConnection();conn.setConnectTimeout(3 * 1000);try (InputStream inStream = conn.getInputStream();OutputStream outputStream = response.getOutputStream();) {byte[] buffer = new byte[1204];while ((byteRead = inStream.read(buffer)) != -1) {outputStream.write(buffer, 0, byteRead);}outputStream.flush();}} catch (Exception e) {log.warn("download下载失败!", e);}}

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Mybatis 返回 Map 对象

一、场景介绍 假设有如下一张学生表&#xff1a; CREATE TABLE student (id int NOT NULL AUTO_INCREMENT COMMENT 主键,name varchar(100) NOT NULL COMMENT 姓名,gender varchar(10) NOT NULL COMMENT 性别,grade int NOT NULL COMMENT 年级,PRIMARY KEY (id) ) ENGINEInnoD…

【C++篇】引领C++模板初体验:泛型编程的力量与妙用

文章目录 C模板编程前言第一章: 初始模板与函数模版1.1 什么是泛型编程&#xff1f;1.1.1 为什么要有泛型编程&#xff1f;1.1.1 泛型编程的优势 1.2 函数模板的基础1.2.1 什么是函数模板&#xff1f;1.2.2 函数模板的定义格式1.2.3 示例&#xff1a;通用的交换函数输出示例&am…

华为HarmonyOS地图服务 11 - 如何在地图上增加点注释?

场景介绍 本章节将向您介绍如何在地图的指定位置添加点注释以标识位置、商家、建筑等&#xff0c;并可以通过信息窗口展示详细信息。 点注释支持功能&#xff1a; 支持设置图标、文字、碰撞规则等。支持添加点击事件。 PointAnnotation有默认风格&#xff0c;同时也支持自定…

文献阅读(220)MRCN

题目&#xff1a;MRCN: Throughput-Oriented Multicast Routing for Customized Network-on-Chips时间&#xff1a;2023期刊&#xff1a;TPDS研究机构&#xff1a;韩国成均馆大学 这篇论文探讨的问题是多播死锁问题&#xff0c;下图中Packet A分成两条路径&#xff0c;但在rou…

Leetcode—1014. 最佳观光组合【中等】

2024每日刷题&#xff08;164&#xff09; Leetcode—1014. 最佳观光组合 实现代码 class Solution { public:int maxScoreSightseeingPair(vector<int>& values) {int mxPre values[0] 0;int ans 0;for(int i 1; i < values.size(); i) {ans max(ans, mxP…

python绘制弦图-科研作图

一、背景 弦图以其直观、精美的展示方式受到越来越多人的关注&#xff0c;它不仅能够有效展示两个变量之间的联系&#xff0c;还能同时展现多个变量间的复杂互动&#xff0c;本文将通过Python语言中的pycirclize库&#xff0c;带你深入了解如何绘制弦图。 弦图是一种圆…

51单片机——矩阵键盘

一、矩阵键盘原理图 我们发现: P17,P16,P15,P14控制行&#xff0c; P13,P12,P11,P10控制列。 所以我们如果要选择第四列&#xff0c;只需要把整个P1先给高电位1&#xff0c;再把P10给低电位0。 二、代码 P10xFF; P100; if(P170){Delay(20);while(P170);Delay(20);KeyNum…

计算机毕业设计python+spark知识图谱房价预测系统 房源推荐系统 房源数据分析 房源可视化 房源大数据大屏 大数据毕业设计 机器学习

《PythonSpark知识图谱房价预测系统》开题报告 一、研究背景与意义 随着城市化进程的加速和房地产市场的不断发展&#xff0c;房价成为影响人们生活质量的重要因素之一。准确预测房价不仅有助于政府制定科学的房地产政策&#xff0c;还能为开发商提供市场参考&#xff0c;同时…

DriveMatriX Highway Dataset :高速公路驾驶数据集(猫脸码客 第196期)

DriveMatriX Highway Dataset 1.0&#xff1a;自动驾驶与ADAS感知验证的里程碑 在当今快速发展的自动驾驶&#xff08;AV&#xff09;和高级驾驶辅助系统&#xff08;ADAS&#xff09;领域&#xff0c;数据的获取与处理成为了推动技术进步的关键因素。为了在这些复杂且多变的交…

【软件测试】Bug 篇

哈喽&#xff0c;哈喽&#xff0c;大家好~ 我是你们的老朋友&#xff1a;保护小周ღ 今天给大家带来的是 【软件测试】Bug 篇&#xff0c;首先了解, 什么是Bug, 如何定义一个Bug, 如何描述一个 Bug, Bug的级别, 和 Bug 的生命周期, 以及测试人员跟开发人员产生争执如何处理,…

【Linux】常用指令【更详细,带实操】

Linux全套讲解系列&#xff0c;参考视频-B站韩顺平&#xff0c;本文的讲解更为详细 目录 一、文件目录指令 1、cd【change directory】指令 ​ 2、mkdir【make dir..】指令​ 3、cp【copy】指令 ​ 4、rm【remove】指令 5、mv【move】指令 6、cat指令和more指令 7、less和…

【设计模式】万字详解:深入掌握五大基础行为模式

作者&#xff1a;后端小肥肠 &#x1f347; 我写过的文章中的相关代码放到了gitee&#xff0c;地址&#xff1a;xfc-fdw-cloud: 公共解决方案 &#x1f34a; 有疑问可私信或评论区联系我。 &#x1f951; 创作不易未经允许严禁转载。 姊妹篇&#xff1a; 【设计模式】&#xf…

多模态大模型应用开发技术学习

前篇提到多模态模型应用是未来的应用方向&#xff0c;本篇就聊聊技术学习方面的内容。 应用场景 多模态大模型技术的应用场景非常广泛&#xff0c;涵盖了从日常生活到专业领域的各个方面。以下是一些主要的应用场景&#xff1a; 办公自动化&#xff1a;多模态大模型可以用于…

计算机网络-小型综合网络的搭建涉及到无线路由交换安全

目录 1 拓扑架构 2 做项目的思路 3 做配置 3.1先做核心交换 3.2 防火墙的配置 4 ac 和ap 的配置 4.1 ac上配置安全的东西 5.1 测试​编辑 1 拓扑架构 要求看上面的图 2 做项目的思路 这张网很明显是一个小综合&#xff0c;设计到我们的无线交换&#xff0c;路由…

jdk11特性介绍

JDK 11&#xff08;也称为Java 11&#xff09;是Java平台的一个重要版本&#xff0c;它引入了许多新特性和改进&#xff0c;旨在提高开发者的生产力和Java平台的性能。以下是一些JDK 11的主要特性&#xff1a; 局部变量类型推断&#xff08;Local-Variable Syntax for Lambda P…

linux中vim编辑器的应用实例

前言 Linux有大量的配置文件&#xff0c;其中编辑一些配置文件&#xff0c;最常用的工具就是 Vim &#xff0c;本文介绍一个实际应用的Vim编辑器开发文档的实例。 Vim是一个类似于Vi的著名的功能强大、高度可定制的文本编辑器&#xff0c;在Vi的基础上改进和增加了很多特性。…

【C语言零基础入门篇 - 17】:排序算法

文章目录 排序算法排序的基本概念冒泡排序选择排序插入排序 排序算法 排序的基本概念 1、什么是排序&#xff1f; 排序是指把一组数据以某种关系&#xff08;递增或递减&#xff09;按顺序排列起来的一种算法。 例如&#xff1a;数列 8、3、5、6、2、9、1、0、4、7 递增排序…

如何防止SQL注入

目录 SQL注入 1、 什么是SQL注入 2、 避免SQL注入 PreparedStatement【重点】 SQL注入 1、 什么是SQL注入 select * from tb_user where username 111 and password 111select * from tb_user where username 111 and password 111 or 11 用户输入的数据中有SQL关键词…

java日志框架之Log4j

文章目录 一、Log4j简介二、Log4j组件介绍1、Loggers (日志记录器)2、Appenders&#xff08;输出控制器&#xff09;3、Layout&#xff08;日志格式化器&#xff09; 三、Log4j快速入门四、Log4j自定义配置文件输出日志1、输出到控制台2、输出到文件3、输出到数据库 五、Log4j自…

【HTTPS】中间人攻击和证书的验证

中间人攻击 服务器可以创建出一堆公钥和私钥&#xff0c;黑客也可以按照同样的方式&#xff0c;创建一对公钥和私钥&#xff0c;冒充自己是服务器&#xff08;搅屎棍&#xff09; 黑客自己也能生成一对公钥和私钥。生成公钥和私钥的算法是开放的&#xff0c;服务器能生产&…