|
|
|
@ -1,5 +1,6 @@
|
|
|
|
|
package com.logpm.report.util; |
|
|
|
|
|
|
|
|
|
import cn.hutool.core.collection.CollUtil; |
|
|
|
|
import cn.hutool.core.collection.CollectionUtil; |
|
|
|
|
import com.alibaba.excel.EasyExcel; |
|
|
|
|
import com.alibaba.excel.ExcelWriter; |
|
|
|
@ -8,18 +9,27 @@ import com.baomidou.mybatisplus.core.conditions.Wrapper;
|
|
|
|
|
import com.baomidou.mybatisplus.extension.plugins.pagination.Page; |
|
|
|
|
import com.logpm.report.constant.ReportConstants; |
|
|
|
|
import com.logpm.report.service.ExportReader; |
|
|
|
|
import com.logpm.report.service.IAsyncService; |
|
|
|
|
import lombok.extern.slf4j.Slf4j; |
|
|
|
|
import org.apache.poi.ss.formula.functions.T; |
|
|
|
|
import org.jetbrains.annotations.NotNull; |
|
|
|
|
import org.springblade.core.log.exception.ServiceException; |
|
|
|
|
import org.springframework.beans.factory.annotation.Autowired; |
|
|
|
|
import org.springframework.stereotype.Component; |
|
|
|
|
|
|
|
|
|
import javax.annotation.Resource; |
|
|
|
|
import javax.servlet.http.HttpServletResponse; |
|
|
|
|
import java.io.IOException; |
|
|
|
|
import java.io.UnsupportedEncodingException; |
|
|
|
|
import java.net.URLEncoder; |
|
|
|
|
import java.nio.charset.StandardCharsets; |
|
|
|
|
import java.util.HashMap; |
|
|
|
|
import java.util.ArrayList; |
|
|
|
|
import java.util.List; |
|
|
|
|
import java.util.Map; |
|
|
|
|
import java.util.concurrent.CompletableFuture; |
|
|
|
|
import java.util.concurrent.ConcurrentHashMap; |
|
|
|
|
import java.util.concurrent.CountDownLatch; |
|
|
|
|
import java.util.concurrent.Executor; |
|
|
|
|
import java.util.concurrent.ExecutorService; |
|
|
|
|
|
|
|
|
|
/** |
|
|
|
@ -28,6 +38,8 @@ import java.util.concurrent.ExecutorService;
|
|
|
|
|
* @author zhaoqiaobo |
|
|
|
|
* @create 2024-03-06 17:48 |
|
|
|
|
*/ |
|
|
|
|
@Slf4j |
|
|
|
|
@Component |
|
|
|
|
public class ReportExcelUtil { |
|
|
|
|
|
|
|
|
|
public static final String CONTENT_TYPE = "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet"; |
|
|
|
@ -41,6 +53,11 @@ public class ReportExcelUtil {
|
|
|
|
|
*/ |
|
|
|
|
public static final Long PAGESIZE = 100000L; |
|
|
|
|
|
|
|
|
|
@Resource |
|
|
|
|
private IAsyncService asyncService; |
|
|
|
|
@Autowired |
|
|
|
|
private Executor asyncExecutor; |
|
|
|
|
|
|
|
|
|
/** |
|
|
|
|
* 设置excel导出的返回头 |
|
|
|
|
* |
|
|
|
@ -88,7 +105,7 @@ public class ReportExcelUtil {
|
|
|
|
|
ReportExcelUtil.setExportHeader(response, fileName); |
|
|
|
|
// 获取需要导出的总条数
|
|
|
|
|
Long count = service.getCount(query); |
|
|
|
|
if(count > 500000L){ |
|
|
|
|
if (count > 500000L) { |
|
|
|
|
throw new ServiceException("导出数据不能大于50w,请增加检索条件"); |
|
|
|
|
} |
|
|
|
|
// 每个 sheet 放多少条数据
|
|
|
|
@ -98,7 +115,7 @@ public class ReportExcelUtil {
|
|
|
|
|
// 总共需要查询数据库几次
|
|
|
|
|
int findNum = (int) Math.ceil((count / pageSize.doubleValue())); |
|
|
|
|
// 封装查询后需要导出的数据
|
|
|
|
|
Map<Integer, List<Object>> pageMap = new HashMap<>(ReportConstants.HASHMAP_DEFAULT_SIZE); |
|
|
|
|
Map<Integer, List<Object>> pageMap = new ConcurrentHashMap<>(ReportConstants.HASHMAP_DEFAULT_SIZE); |
|
|
|
|
// 定义线程池,大表导出使用一个线程池
|
|
|
|
|
ExecutorService executorService = ThreadPoolUtil.getThreadPool(); |
|
|
|
|
// 控制线程全部查询完后执行导出操作
|
|
|
|
@ -111,6 +128,7 @@ public class ReportExcelUtil {
|
|
|
|
|
e.printStackTrace(); |
|
|
|
|
} |
|
|
|
|
int num = 0; |
|
|
|
|
List<CompletableFuture<List>> futures = new ArrayList<>(); |
|
|
|
|
for (int i = 0; i < sheetNum; i++) { |
|
|
|
|
// sheet编号
|
|
|
|
|
int finalI = i + 1; |
|
|
|
@ -123,28 +141,37 @@ public class ReportExcelUtil {
|
|
|
|
|
// 查询次数+1
|
|
|
|
|
num = num + 1; |
|
|
|
|
// 提交线程执行查询
|
|
|
|
|
executorService.submit(() -> { |
|
|
|
|
// 剩余数据
|
|
|
|
|
long remaindNum = count - (finalNum * pageSize); |
|
|
|
|
// 当前第几次查询
|
|
|
|
|
int current = finalNum + 1; |
|
|
|
|
// 当前查询需要查询几条数据
|
|
|
|
|
long lastNum = remaindNum > pageSize ? pageSize : remaindNum; |
|
|
|
|
// 分页查询对象
|
|
|
|
|
Page<T> page = new Page(current, lastNum); |
|
|
|
|
// 执行查询操作返回数据
|
|
|
|
|
List deliveryTrainPage = service.findList(page, query); |
|
|
|
|
// 获取组装的数据列表
|
|
|
|
|
List<Object> detailVo = pageMap.get(finalI); |
|
|
|
|
// 将查询数据添加到本地缓存
|
|
|
|
|
if (CollectionUtil.isEmpty(detailVo)) { |
|
|
|
|
pageMap.put(finalI, deliveryTrainPage); |
|
|
|
|
} else { |
|
|
|
|
pageMap.get(finalI).addAll(deliveryTrainPage); |
|
|
|
|
} |
|
|
|
|
// 当前线程执行完成
|
|
|
|
|
countDownLatch.countDown(); |
|
|
|
|
}); |
|
|
|
|
// CompletableFuture<List> future = asyncService.getListCompletableFuture(service, query, pageSize, count, finalNum, pageMap, finalI);
|
|
|
|
|
// CompletableFuture<List> future = CompletableFuture.supplyAsync(() -> {
|
|
|
|
|
// // 剩余数据
|
|
|
|
|
// long remaindNum = count - (finalNum * pageSize);
|
|
|
|
|
// // 当前第几次查询
|
|
|
|
|
// int current = finalNum + 1;
|
|
|
|
|
// // 当前查询需要查询几条数据
|
|
|
|
|
// long lastNum = remaindNum > pageSize ? pageSize : remaindNum;
|
|
|
|
|
// // 分页查询对象
|
|
|
|
|
// Page<T> page = new Page(current, lastNum);
|
|
|
|
|
// // 执行查询操作返回数据
|
|
|
|
|
// return service.findList(page, query);
|
|
|
|
|
// }, asyncExecutor);
|
|
|
|
|
// future.thenAccept(list -> {
|
|
|
|
|
// if(CollUtil.isNotEmpty(list)){
|
|
|
|
|
// // 获取组装的数据列表
|
|
|
|
|
// List<Object> detailVo = pageMap.get(finalI);
|
|
|
|
|
// // 将查询数据添加到本地缓存
|
|
|
|
|
// if (CollectionUtil.isEmpty(detailVo)) {
|
|
|
|
|
// pageMap.put(finalI, list);
|
|
|
|
|
// } else {
|
|
|
|
|
// pageMap.get(finalI).addAll(list);
|
|
|
|
|
// }
|
|
|
|
|
// }
|
|
|
|
|
// });
|
|
|
|
|
// futures.add(future);
|
|
|
|
|
// asyncExecutor.execute(() -> {
|
|
|
|
|
|
|
|
|
|
asyncService.initExportData(service, query, pageSize, count, finalNum, pageMap, finalI, countDownLatch); |
|
|
|
|
|
|
|
|
|
// });
|
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
try { |
|
|
|
@ -153,6 +180,13 @@ public class ReportExcelUtil {
|
|
|
|
|
} catch (InterruptedException e) { |
|
|
|
|
e.printStackTrace(); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// try {
|
|
|
|
|
// CompletableFuture<Void> allFutures = CompletableFuture.allOf(futures.toArray(new CompletableFuture[0]));
|
|
|
|
|
// allFutures.join();
|
|
|
|
|
// } catch (Exception e) {
|
|
|
|
|
// log.error("系统异常:{}", e);
|
|
|
|
|
// }
|
|
|
|
|
// 分sheet向excel中写数据,
|
|
|
|
|
for (Map.Entry<Integer, List<Object>> entry : pageMap.entrySet()) { |
|
|
|
|
Integer key = entry.getKey(); |
|
|
|
@ -164,4 +198,54 @@ public class ReportExcelUtil {
|
|
|
|
|
excelWriter.finish(); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
private static void initExportData(ExportReader service, Wrapper query, Long pageSize, Long count, int finalNum, Map<Integer, List<Object>> pageMap, int finalI) { |
|
|
|
|
// 剩余数据
|
|
|
|
|
long remaindNum = count - (finalNum * pageSize); |
|
|
|
|
// 当前第几次查询
|
|
|
|
|
int current = finalNum + 1; |
|
|
|
|
// 当前查询需要查询几条数据
|
|
|
|
|
long lastNum = remaindNum > pageSize ? pageSize : remaindNum; |
|
|
|
|
// 分页查询对象
|
|
|
|
|
Page<T> page = new Page(current, lastNum); |
|
|
|
|
// 执行查询操作返回数据
|
|
|
|
|
List deliveryTrainPage = service.findList(page, query); |
|
|
|
|
// 获取组装的数据列表
|
|
|
|
|
List<Object> detailVo = pageMap.get(finalI); |
|
|
|
|
// 将查询数据添加到本地缓存
|
|
|
|
|
if (CollectionUtil.isEmpty(detailVo)) { |
|
|
|
|
pageMap.put(finalI, deliveryTrainPage); |
|
|
|
|
} else { |
|
|
|
|
pageMap.get(finalI).addAll(deliveryTrainPage); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
private static @NotNull CompletableFuture<List> getListCompletableFuture(ExportReader service, Wrapper query, Long pageSize, Long count, int finalNum, Map<Integer, List<Object>> pageMap, int finalI) { |
|
|
|
|
CompletableFuture<List> future = CompletableFuture.supplyAsync(() -> { |
|
|
|
|
// 剩余数据
|
|
|
|
|
long remaindNum = count - (finalNum * pageSize); |
|
|
|
|
// 当前第几次查询
|
|
|
|
|
int current = finalNum + 1; |
|
|
|
|
// 当前查询需要查询几条数据
|
|
|
|
|
long lastNum = remaindNum > pageSize ? pageSize : remaindNum; |
|
|
|
|
// 分页查询对象
|
|
|
|
|
Page<T> page = new Page(current, lastNum); |
|
|
|
|
// 执行查询操作返回数据
|
|
|
|
|
return service.findList(page, query); |
|
|
|
|
|
|
|
|
|
}); |
|
|
|
|
future.thenAccept(list -> { |
|
|
|
|
if (CollUtil.isNotEmpty(list)) { |
|
|
|
|
// 获取组装的数据列表
|
|
|
|
|
List<Object> detailVo = pageMap.get(finalI); |
|
|
|
|
// 将查询数据添加到本地缓存
|
|
|
|
|
if (CollectionUtil.isEmpty(detailVo)) { |
|
|
|
|
pageMap.put(finalI, list); |
|
|
|
|
} else { |
|
|
|
|
pageMap.get(finalI).addAll(list); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
}); |
|
|
|
|
return future; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
} |
|
|
|
|