Springboot POI导出Excel(浏览器)
作者:多好才叫好。 发布时间:2022-07-08 18:41:17
标签:Springboot,POI,Excel
本文实例为大家分享了Springboot POI导出Excel的具体代码,供大家参考,具体内容如下
需求:页面根据查询条件导出(浏览器)
由于本次导出数据量较大,这里采用XSSFWorkbook多线程进行导出,注:XSSFWorkbook导出excel文件结尾为:“.xlsx”。
导出不需要返回,如有返回则会报异常!
//Controller
@RequestMapping("/stateExport")
public void stateExport(HttpServletResponse response,@RequestParam("deviceId") Long deviceId, String startTime,String endTime) {
try {
deviceMonitorService.stateExport(response, deviceId, startTime,endTime);
LOG.info("导出成功");
} catch (Exception e) {
LOG.error("导出异常:",e.getMessage());
}
}
//Service
@Override
public void stateExport(HttpServletResponse response, Long deviceId, String startTime, String endTime) throws Exception{
//list自己查询得出
List<StateDetailsResult> list = queryStateDetails(deviceId, startTime, endTime);
String time = TimeUtils.YEAR_DAY_SECOND_FORMAT.format(new Date());
//sheet名称
String sheetName = deviceDtls.getName() + "状态"+time;
//文件名称
String excelName = deviceDtls.getName() + "状态"+time+".xlsx";
//文件头
String[] strings = {"状态名称","开始时间","结束时间","状态时长"};
String path = this.getClass().getResource("").getPath() + "excel";
DownloadFileUtil.createDirs(path);
String filePath = path + "/" + sheetName + ".xls";
stateCreateExcel(list,strings,sheetName,excelName,filePath);
DownloadFileUtil.download(filePath, response);
}
public String stateCreateExcel(List<StateDetailsResult> list, String[] strArray,String sheetName,String excelName,String filePath)throws Exception {
// 总数据条数
int dataSize = list.size();
// 线程数
int threadNum = 2;
int threadSize = dataSize / threadNum;
ExecutorService exec = Executors.newFixedThreadPool(threadNum);
//cutList 和输入list类型保持一致
List<StateDetailsResult> cutList = null;
// 第一步,创建一个webbook,对应一个Excel文件
XSSFWorkbook wb = new XSSFWorkbook();
// 第二步,在webbook中添加一个sheet,对应Excel文件中的sheet
XSSFSheet sheet = wb.createSheet(sheetName);
sheet.setDefaultColumnWidth(20);// 默认列宽
// 第三步,在sheet中添加表头第0行,注意老版本poi对Excel的行数列数有限制short
XSSFRow row = sheet.createRow((int) 0);
// 第四步,创建单元格,并设置值表头 设置表头居中
XSSFCellStyle style = wb.createCellStyle();
// 创建一个居中格式
style.setAlignment(HSSFCellStyle.ALIGN_CENTER);
// 添加excel title
XSSFCell cell = null;
for (int i = 0; i < strArray.length; i++) {
cell = row.createCell((short) i);
cell.setCellValue(strArray[i]);
cell.setCellStyle(style);
}
List<Callable<Integer>> tasks = new ArrayList<Callable<Integer>>();
Callable<Integer> task = null;
// 第五步,写入实体数据 实际应用中这些数据从数据库得到,list中字符串的顺序必须和数组strArray中的顺序一致
int startNum ;
System.out.println("任务开始,总数:"+list.size());
// 开始时间
long start = System.currentTimeMillis();
System.out.println("线程任务执行");
for (int i = 0; i < threadNum; i++) {
startNum = threadSize * i;
if (i == threadNum - 1) {
cutList = list.subList(threadSize * i, dataSize);
} else {
cutList = list.subList(threadSize * i, threadSize * (i + 1));
}
//listStr 和输入list类型保持一致
final List<StateDetailsResult> listStr = cutList;
int finalStartNum = startNum;
task = new Callable<Integer>() {
final int s= finalStartNum;
@Override
public Integer call() throws Exception {
for(int j=0;j<listStr.size();j++){
XSSFRow row =getRow(sheet,s+j);
//设置每一列展示内容自己设置
row.createCell(0).setCellValue(listStr.get(j).getName());
row.createCell(1).setCellValue(listStr.get(j).getStartDateTime());
row.createCell(2).setCellValue(listStr.get(j).getEndDateTime());
row.createCell(3).setCellValue(listStr.get(j).getDateTime());
}
return 1;
}
};
// 这里提交的任务容器列表和返回的Future列表存在顺序对应的关系
tasks.add(task);
}
try {
List<Future<Integer>> results = exec.invokeAll(tasks);
} catch (Exception e) {
e.printStackTrace();
}
// 关闭线程池
exec.shutdown();
DownloadFileUtil.delfile(filePath);
// 第六步,将文件存到指定位置
try {
FileOutputStream fout = new FileOutputStream(filePath);
wb.write(fout);
fout.flush();
fout.close();
} catch (Exception e) {
e.printStackTrace();
}
System.out.println("线程任务执行结束");
System.err.println("执行任务消耗了 :" + (System.currentTimeMillis() - start) + "毫秒");
return filePath;
}
//线程同步,保证不会多插入数据
private synchronized XSSFRow getRow(XSSFSheet sheet, int rownum) {
//如果不包含列头,+1去掉即可
return sheet.createRow(rownum+1);
}
最后附上使用的工具类
package com.sec.deviceband.utils;
import javax.servlet.http.HttpServletResponse;
import java.io.*;
import java.net.URLEncoder;
public class DownloadFileUtil {
/**
* 判断路径是否存在不存在则创建
* @param path
*/
public static void createDirs(String path) {
File file = new File(path);
if (!file.exists()) {
file.mkdirs();
}
}
/**
* 下载
*
* @param path
* @param response
*/
public static void download(String path, HttpServletResponse response) {
try {
// path是指欲下载的文件的路径。
File file = new File(path);
// 取得文件名。
String filename = file.getName();
// 以流的形式下载文件。
InputStream fis = new BufferedInputStream(new FileInputStream(path));
byte[] buffer = new byte[fis.available()];
fis.read(buffer);
fis.close();
// 清空response
response.reset();
// 设置response的Header
response.addHeader("Content-Disposition", "attachment;filename="
+ URLEncoder.encode(filename, "utf-8"));
response.addHeader("Content-Length", "" + file.length());
OutputStream toClient = new BufferedOutputStream(
response.getOutputStream());
response.setContentType("application/vnd.ms-excel;charset=utf-8");
toClient.write(buffer);
toClient.flush();
toClient.close();
delfile(path);
} catch (IOException ex) {
ex.printStackTrace();
}
}
/**
* 判断文件是否存在则删除
*
* @param filepath
*/
public static void delfile(String filepath) {
File file = new File(filepath);
if (file.exists()) {
file.delete();
}
}
}
测试方式:浏览器输入请求路径
测试效果:
由于水平有限,博客中难免会有一些错误,有纰漏之处恳请各位大佬不吝赐教!
来源:https://blog.csdn.net/weixin_43859696/article/details/106115393


猜你喜欢
- 今天给大家带来的是一块用WPF 实现魔方的小游戏,先看一下效果图 代码如下,先写一个类,用来判断是否可以移动using System;usi
- 抛砖今天使用monio做S3存储时,添加云服务器初始化时一直在构建客户端抛出异常。MinioClient.builder() //NoCla
- Flayway是一款数据库版本控制管理工具,,支持数据库版本自动升级,Migrations可以写成sql脚本,也可以写在java代码里;不仅
- C#函数式程序设计之作用域在C#中,变量的作用域是严格确定的。其本质是所有代码生存在类的方法中、所有变量只生存于声明它们的模块中或者之后的代
- Fragment Android是在Android 3.0 (API level 11)开始引入Fragment的。 可以把Fragment
- 1.继承Thread类,重写run方法2.实现Runnable接口,重写run方法,实现Runnable接口的实现类的实例对象作为Threa
- 一、前言在mybatis官网中,有插件一说 mybatis plugins 如果同时有多个插件,那么他们的执行顺序是怎样的?二、准备工作、代
- 添加记录后获取主键ID,这是一个很常见的需求,特别是在一次前端调用中需要插入多个表的场景。除了添加单条记录时获取主键值,有时候可能需要获取批
- 本文实例讲述了C#读取文本文件到listbox组件的方法。分享给大家供大家参考。具体实现方法如下:private void AddTxtTo
- 本文最终结果大概是这样的,使用java技术随机生成10个数,然后填充一个数组并在消息框中显示数组内容,接着对数组求和输出,将结果显示在消息框
- 这篇文章中我们来继续学习Picasso中还提供了哪些扩展功能,一个好的框架一定是扩展性强的,你需要的我刚好有。下面看一下都提供了哪些扩展功能
- 自定义类:using System;using System.Collections.Generic;using System.Linq;u
- 前言我们经常会被问到这么一个问题:SpringBoot相对于spring有哪些优势呢?其中有一条答案就是SpringBoot自动注入。那么自
- 本文实例讲述了Android开发实现的简单五子棋游戏。分享给大家供大家参考,具体如下:我刚刚在Android上写的一个五子棋的小程序,在这里
- 一、悲观锁和乐观锁1.1. 乐观锁顾名思义,就是很乐观,每次去拿数据的时候都认为别人不会修改,所以不会上锁,但是在更新的时候会判断一下在此期
- 1.结构体类型C语言中的2种类型:原生类型和自定义类型,结构体类型是一种自定义类型。2.结构体使用时先定义结构体类型再用类型定义变量->
- SpringMVC一般使用MultipartFile来做文件的上传,通过MultipartFile的getContentType()方法判定
- 本文实例讲述了C++判断pe文件的方法。分享给大家供大家参考。具体实现方法如下:#include <afxdlgs.h>是为了使
- 在项目中常常常使用到DataTable,假设DataTable使用得当,不仅能使程序简洁有用,并且可以提高性能,达到事半功倍的效果,List
- 这篇文章主要介绍了基于spring security实现登录注销功能过程解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的