功能概述

主要实现的功能:

1.分页查询,避免一次性查询全部数据加载到内存引起频繁FULL GC甚至OOM

2.当数据量超过单个工作簿最大行数(1048575)时,自动将数据写入新的工作簿

3.支持百万级数据量导出

具体实现

第一,定义数据实体父类和分页对象。

// 数据实体父类
public abstract class DataEntity {} // 分页对象
@Data
public class Page<T> {
/** 分页参数,第几页 */
protected int pageNo; /** 分页参数,当前页数量 */
protected int pageSize; /** 数据列表 */
protected List<T> list; /** 数据总量 */
protected long count; public Page(int pageNo, int pageSize) {
this.pageNo = pageNo;
this.pageSize = pageSize;
}
}

第二,既然是分页查询导出,就需要一个实现分页查询的接口。

// 数据分页查询接口
public interface PageQuerier<T> {
/**
* 查询指定页数据,同时会查询总数
*
* @param page
* @param entity
* @return
*/
Page<T> findPage(Page<T> page, DataEntity entity);
/**
* 查询指定页数据,只查询数据,不查询总数
*
* @param page
* @param entity
* @return
*/
List<T> findList(Page<T> page, DataEntity entity);
}

第三,封装分页查询并导出数据的核心服务类。

// 分页导出Excel
public class ExcelExporter<T> {
/** 实现分页查询 */
private PageQuerier<T> pageQuerier; /**
* @param pageQuerier 分页查询服务
*/
public ExcelExporter(PageQuerier<T> pageQuerier) {
this.pageQuerier = pageQuerier;
}
/**
* 分页查询数据并写入到下载输出流
*
* @param response
* @param fileName 文件名,如:测试.xlsx
* @param header 表头映射类
* @param entity 数据实体对象
* @throws IOException
*/
public void write(HttpServletResponse response, String fileName, Class header, DataEntity entity) throws IOException {
if (header == null) {
header = entity.getClass();
}
// 内容样式策略
WriteCellStyle contentWriteCellStyle = new WriteCellStyle();
// 垂直居中,水平居中
contentWriteCellStyle.setVerticalAlignment(VerticalAlignment.CENTER);
contentWriteCellStyle.setHorizontalAlignment(HorizontalAlignment.CENTER);
contentWriteCellStyle.setBorderLeft(BorderStyle.THIN);
contentWriteCellStyle.setBorderTop(BorderStyle.THIN);
contentWriteCellStyle.setBorderRight(BorderStyle.THIN);
contentWriteCellStyle.setBorderBottom(BorderStyle.THIN);
HorizontalCellStyleStrategy contentStyleStrategy = new HorizontalCellStyleStrategy();
contentStyleStrategy.setContentWriteCellStyleList(Arrays.asList(new WriteCellStyle[]{contentWriteCellStyle}));
// 一定要对文件名做URL编码,否则会出现中文乱码
String file = URLEncoder.encode(fileName, StandardCharsets.UTF_8.toString());
response.reset();
response.addHeader("Content-Disposition", "attachment;filename=" + file);
response.addHeader("filename", file);
response.setContentType("application/octet-stream; charset=UTF-8");
ExcelWriter writer = new ExcelWriterBuilder()
.autoCloseStream(true)
.automaticMergeHead(false)
.excelType(ExcelTypeEnum.XLSX)
.file(response.getOutputStream())
.head(header)
.registerWriteHandler(contentStyleStrategy)
.build();
// xlsx文件单个工作簿上限是1048575行,这里如果超过104W需要分Sheet
long sheetMax = 1048575;
// 每一个工作簿可写入的数据量,需要把表头所占的行算上
long sheetTotal = 1;
// 工作簿编号
int sheetNum = 1;
// 分页数
int pageNo = 1;
// 每页查询数量
int pageSize = 50000;
WriteSheet writeSheet = EasyExcel.writerSheet("sheet"+sheetNum).head(header).build();
// 先查询第一页,得到总数
Page<T> page = this.pageQuerier.findPage(new Page(pageNo, pageSize), entity);
writer.write(page.getList(), writeSheet);
if (page.getCount() <= 0 || page.getCount() <= pageSize) {
// 没有数据或只有一页数据
writer.finish();
return;
}
sheetTotal += page.getList().size();
long total = page.getCount();
long fetch = page.getList().size();
for (;;) {
// 只查询分页数据,不查询分页总数
pageNo++;
List<T> list = this.pageQuerier.findList(new Page(pageNo, pageSize), entity);
fetch += list.size();
sheetTotal += list.size();
if (sheetTotal >= sheetMax) {
sheetTotal = 1 + list.size();
sheetNum++;
writeSheet = EasyExcel.writerSheet("sheet"+sheetNum).head(header).build();
}
writer.write(list, writeSheet);
if (fetch >= total) {
writer.finish();
break;
}
}
writer.finish();
}
}

完成上述核心接口和类的定义之后,就可以调用封装好的分页查询和导出服务了。

// 以分页查询数据库方式导出百万级数据
@RequestMapping("/downloadPage")
public String downloadByPage(HttpServletRequest request, HttpServletResponse response,
@RequestParam("name") String name,
@RequestParam("age") int age) throws IOException {
SubjectExport param = SubjectExport.builder().name(name).age(age).build();
String fileName = String.format("%s%s.xlsx", "数据导出", System.currentTimeMillis());
new ExcelExporter<SubjectExport>(this.subjectPageQuerier).write(response, fileName, SubjectExport.class, param);
return null;
}

完整示例代码详见:test-web-downloadtip,可以直接下载到本地运行。

基于EasyExcel实现的分页数据下载封装的更多相关文章

  1. MODIS系列之NDVI(MOD13Q1)一:数据下载(一)基于插件

    引言: 写MODIS数据处理这个系列文章的初衷,主要是为了分享本人处理MODIS数据方面的一些经验.鉴于网上对这方面系统性的总结还比较少,我搜集资料时也是走了许多的弯路,因此希望通过此文让初学者能够更 ...

  2. DRF 返回数据的封装,和分页

    DRF 返回数据的封装,和分页 1 返回值的 封装 自定义一个类,初始化基本的返回数据信息 class BaseResponse(object): """ 初始化基本的返 ...

  3. MODIS系列之NDVI(MOD13Q1)一:数据下载(二)基于FTP

    这一篇我们来介绍下MODIS数据的下载方式.当然这边主要是介绍国外网站的下载方式,国内网站的普遍是在地理空间数据云和遥感集市下载.国外网站(NASA官网)下载方式主要介绍两种.本篇主要针对第一种方式, ...

  4. 基于EasyExcel的大数据量导入并去重

    源码:https://gitee.com/antia11/excel-data-import-demo 背景:客户需要每周会将上传一个 Excel 数据文件,数据量单次为 20W 以上,作为其他模块和 ...

  5. 源码来袭!!!基于jquery的ajax分页插件(demo+源码)

    前几天打开自己的博客园主页,无意间发现自己的园龄竟然有4年之久了.可是看自己的博客列表却是空空如也,其实之前也有写过,但是一直没发布(然而好像并没有什么卵用).刚开始学习编程时就接触到博客园,且在博客 ...

  6. android中RecycleView分页原生代码封装,无任何第三方代

    概述 RecycleView分页加载封装,简单方便,功能齐全 详细 代码下载:http://www.demodashi.com/demo/13283.html 一.场景: 在项目开发中经常使用到列表集 ...

  7. 使用VUE+SpringBoot+EasyExcel 整合导入导出数据

    使用VUE+SpringBoot+EasyExcel 整合导入导出数据 创建一个普通的maven项目即可 项目目录结构 1 前端 存放在resources/static 下 index.html &l ...

  8. 基于SqlSugar的开发框架循序渐进介绍(13)-- 基于ElementPlus的上传组件进行封装,便于项目使用

    在我们实际项目开发过程中,往往需要根据实际情况,对组件进行封装,以更简便的在界面代码中使用,在实际的前端应用中,适当的组件封装,可以减少很多重复的界面代码,并且能够非常简便的使用,本篇随笔介绍基于El ...

  9. 基于vue2.0的分页组件开发

    今天安排的任务是写基于vue2.0的分页组件,好吧,我一开始是觉得超级简单的,但是越写越写不出来,写的最后乱七八糟的都不知道下句该写什么了,所以重新捋了思路,小结一下- 首先写组件需要考虑: 要从父组 ...

随机推荐

  1. MVC 调试页面路径变成 Views/Controller/Action.cshtml问题

    MVC在路由里面已经写好了路径,但是调试时地址栏还是会变成 Views/Controller/Action.cshtml,导致报404错误,找不到路径. 原因可能是你将某一页面设为了起始页,导致每次运 ...

  2. WPF|快速添加新手引导功能(支持MVVM)

    阅读导航 前言 案例一 案例二 案例三(本文介绍的方式) 如何使用? 控件如何开发的? 总结 1. 前言 案例一 站长分享过 眾尋 大佬的一篇 WPF 简易新手引导 一文,新手引导的效果挺不错的,如下 ...

  3. CF1580E Railway Construction

    CF1580E Railway Construction 铁路系统中有 \(n\) 个车站和 \(m\) 条双向边,有边权,无重边.这些双向边使得任意两个车站互相可达. 你现在要加一些单向边 \((u ...

  4. Web 前端实战(三):雷达图

    前言 在<Canvas 线性图形(五):多边形>实现了绘制多边形的函数.本篇文章将记录如何绘制雷达图.最终实现的效果是这样的: 绘制雷达图 雷达图里外层 如动图中所示,雷达图从里到外一共有 ...

  5. SpringBoot Restful 接口实现

    目录 SpringBoot 核心注解 SpringBoot Restful 接口实现 封装响应数据 SpringBoot 核心注解 SpringBoot 基础入门 注解 说明 Component 声明 ...

  6. 《HALCON数字图像处理》第四章笔记

    目录 第四章 HALCON数据结构 HALCON Image图像 图像通道 HALCON Region区域 Region的初步介绍 Region的点与线 Region的行程 Region的区域特征 H ...

  7. JAVA - error(错误)和exception(异常)有什么区别?

    JAVA - error(错误)和exception(异常)有什么区别? error 表示恢复不是不可能但很困难的情况下的一种严重问题.比如说内存溢出.不可能指望程序能处理这样的情况. excepti ...

  8. LVS+keepalived高可用

    1.keeplived相关 1.1工作原理 Keepalived 是一个基于VRRP协议来实现的LVS服务高可用方案,可以解决静态路由出现的单点故障问题. 在一个LVS服务集群中通常有主服务器(MAS ...

  9. Groovy基础语法

    Groovy 基础语法 变量定义 1.支持动态类型,使用def关键字定义变量 // Java中定义变量的方式 int age = 18; String name = "张三"; / ...

  10. numpy中shape的部分解释

    转载自:https://blog.csdn.net/qq_28618765/article/details/78081959和https://www.jianshu.com/p/e083512e4f4 ...