Springboot POI导出Excel(浏览器)

 更新时间:2020年5月17日 00:05  点击:1670

本文实例为大家分享了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();
 }
 }
}

测试方式:浏览器输入请求路径

测试效果:

由于水平有限,博客中难免会有一些错误,有纰漏之处恳请各位大佬不吝赐教!

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持猪先飞。

[!--infotagslink--]

相关文章

  • 解决springboot使用logback日志出现LOG_PATH_IS_UNDEFINED文件夹的问题

    这篇文章主要介绍了解决springboot使用logback日志出现LOG_PATH_IS_UNDEFINED文件夹的问题,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2021-04-28
  • SpringBoot实现excel文件生成和下载

    这篇文章主要为大家详细介绍了SpringBoot实现excel文件生成和下载,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...2021-02-09
  • c#读取excel方法实例分析

    这篇文章主要介绍了c#读取excel方法,实例分析了C#读取excel文件的原理与相关技巧,需要的朋友可以参考下...2020-06-25
  • 详解springBoot启动时找不到或无法加载主类解决办法

    这篇文章主要介绍了详解springBoot启动时找不到或无法加载主类解决办法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2020-09-16
  • SpringBoot集成Redis实现消息队列的方法

    这篇文章主要介绍了SpringBoot集成Redis实现消息队列的方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2021-02-10
  • Python导入数值型Excel数据并生成矩阵操作

    这篇文章主要介绍了Python导入数值型Excel数据并生成矩阵操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2020-06-09
  • 解决Springboot get请求是参数过长的情况

    这篇文章主要介绍了解决Springboot get请求是参数过长的情况,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2020-09-17
  • Spring Boot项目@RestController使用重定向redirect方式

    这篇文章主要介绍了Spring Boot项目@RestController使用重定向redirect方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教...2021-09-02
  • Springboot+TCP监听服务器搭建过程图解

    这篇文章主要介绍了Springboot+TCP监听服务器搭建过程,本文通过图文并茂的形式给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下...2020-10-28
  • C# winform打开Excel文档的方法总结(必看篇)

    下面小编就为大家带来一篇C# winform打开Excel文档的方法总结(必看篇)。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧...2020-06-25
  • springBoot 项目排除数据库启动方式

    这篇文章主要介绍了springBoot 项目排除数据库启动方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教...2021-09-10
  • C# 导出Excel的6种简单方法实现

    C# 导出 Excel 的6种简单方法:数据表导出到 Excel,对象集合导出到 Excel,数据库导出到 Excel,微软网格控件导出到 Excel,数组导出到 Excel,CSV 导出到 Excel,你都会了吗?需要的朋友们下面随着小编来一起学习学习吧...2020-06-25
  • 详解SpringBoot之访问静态资源(webapp...)

    这篇文章主要介绍了详解SpringBoot之访问静态资源(webapp...),文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2020-09-14
  • SpringBoot接口接收json参数解析

    这篇文章主要介绍了SpringBoot接口接收json参数解析,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教...2021-10-19
  • C#实现Excel表数据导入Sql Server数据库中的方法

    这篇文章主要介绍了C#实现Excel表数据导入Sql Server数据库中的方法,结合实例形式详细分析了C#读取Excel表数据及导入Sql Server数据库的具体操作步骤与相关操作技巧,需要的朋友可以参考下...2020-06-25
  • springboot中使用@Transactional注解事物不生效的坑

    这篇文章主要介绍了springboot中使用@Transactional注解事物不生效的原因,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2021-01-26
  • springboot多模块包扫描问题的解决方法

    这篇文章主要介绍了springboot多模块包扫描问题的解决方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2020-09-16
  • C#实现将数据导出到word或者Excel中的方法

    这篇文章主要介绍了C#实现将数据导出到word或者Excel中的方法,涉及C#操作word及Excel格式文件的方法,具有一定参考借鉴价值,需要的朋友可以参考下...2020-06-25
  • Springboot mybatis plus druid多数据源解决方案 dynamic-datasource的使用详解

    这篇文章主要介绍了Springboot mybatis plus druid多数据源解决方案 dynamic-datasource的使用,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下...2020-11-18
  • Springboot+MDC+traceId日志中打印唯一traceId

    本文主要介绍了Springboot+MDC+traceId日志中打印唯一traceId,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...2021-10-17