【webFlux Excel导出插入图片】POI导出Excel的时候插入图片
- 工作小总结
- 时间:2023-03-05 14:08
- 16318人已阅读
简介
在工作中,有时候,我们会遇到这样的需求:导出数据的时候,同时把图片导出。本文使用于WebFlux响应式开发导出Excel功能。SpringMVC模式可以在此基础上做少许修改即可实现在SpringMVC模式导出带有图片的需求如下:导出效果:代码如下:入口:public Mono<Void> exportHistoryReport(HistoryDataParam&nb
🔔🔔🔔好消息!好消息!🔔🔔🔔
有需要的朋友👉:联系凯哥
在工作中,有时候,我们会遇到这样的需求:导出数据的时候,同时把图片导出。
本文使用于WebFlux 响应式开发导出Excel功能。SpringMVC模式可以在此基础上做少许修改即可实现在SpringMVC模式导出带有图片的
需求如下:
导出效果:
代码如下:
入口:
public Mono<Void> exportHistoryReport(HistoryDataParam historyDataParam, ServerHttpResponse response,Integer fromType) throws Exception { String dimensionType = historyDataParam.getDimensionType(); String quotaType = historyDataParam.getQuotaType(); String image = ""; String picBase64 = historyDataParam.getPicBase64(); if(StringUtils.isEmpty(picBase64)){ picBase64 = image; } /* * 数据样式: * /8O+dSe8hCSlAaKJ0BBEQFBBQseFiwUVkURHEgtjWdV0bLPpTV10BRYodsWH5i64guIiAqKyASu8lgfTeJzNz/8 ........ * 图片编码只要‘base64,’后面的部分 * 图片数据通过表单传递到了后台,可以在后台获取到信息。但是有一点必须注意:数据中的 "+"加号会因为传递变为 " "空格,所以需要替换一下 */ String dataChartCostStr = (URLDecoder.decode(picBase64.substring(22), "UTF-8")).replaceAll(" ", "+"); Mono<List<HistoryQuotaBaseResult>> dataMono= Mono.empty(); String tableName = "数据报表(" + dimensionType + "_" + quotaType + ")"; String[] heards = heardMapping.get("11"); String[] fields = heardMapping.get("13field"); if(1 == fromType){ dataMono = mockDataMono(historyDataParam); if ("2".equals(dimensionType)) { heards = heardMapping.get("12"); } }else{ dataMono = mockDevicePropertyDataMono(historyDataParam); tableName = "数据报表(" + dimensionType + "_" + quotaType + ")"; if ("2".equals(dimensionType)) { heards = heardMapping.get("22"); }else { heards = heardMapping.get("21"); } } List<HistoryQuotaBaseResult> dataList = Lists.newArrayList(); dataMono.map(ms -> { dataList.addAll(ms); return dataList; }).defaultIfEmpty(new ArrayList<>()).subscribe(); return HistoryExcelUtil.getHSSFWorkbook(tableName, tableName + ".xls", heards, fields, dataList, response, dataChartCostStr); }
代码片2:
/** *表头1 */ private static final String[] WEATHER_QUOTA_DAY_TITLE = new String[]{"时间", "气象指标.风速", "气象指标.气温", "气象指标.湿度" , "气象指标.对标数据(去年同期).风速", "气象指标.对标数据(去年同期).气温", "气象指标.对标数据(去年同期).湿度"}; /** * 表头1对应的属性字段 */ private static final String[] WEATHER_QUOTA_INFO_FIELD = new String[]{"dataTime", "windSpeed", "temperature", "humidity" , "windSpeedBenchmark", "temperatureBenchmark", "humidityBenchmark"}; private static Map<String, String[]> heardMapping; static { heardMapping = new HashMap<>(); heardMapping.put("11", WEATHER_QUOTA_HOURS_TITLE); heardMapping.put("13field", WEATHER_QUOTA_INFO_FIELD); }
导出的工具类:
t com.google.common.base.Strings; import org.apache.commons.collections.CollectionUtils; import org.apache.poi.hssf.usermodel.*; import org.apache.poi.hssf.util.HSSFColor; import org.apache.poi.ss.usermodel.CellStyle; import org.apache.poi.ss.usermodel.Font; import org.springframework.core.io.buffer.DataBuffer; import org.springframework.core.io.buffer.DefaultDataBuffer; import org.springframework.core.io.buffer.DefaultDataBufferFactory; import org.springframework.http.HttpHeaders; import org.springframework.http.MediaType; import org.springframework.http.ZeroCopyHttpOutputMessage; import org.springframework.http.server.reactive.ServerHttpResponse; import reactor.core.publisher.Flux; import reactor.core.publisher.FluxSink; import reactor.core.publisher.Mono; import sun.misc.BASE64Decoder; import javax.imageio.ImageIO; import java.awt.image.BufferedImage; import java.io.ByteArrayInputStream; import java.io.ByteArrayOutputStream; import java.io.IOException; import java.io.OutputStream; import java.lang.reflect.Method; import java.net.URLEncoder; import java.nio.charset.StandardCharsets; import java.text.SimpleDateFormat; import java.util.Date; import java.util.List; import java.util.Objects; /** * @author 凯哥Java * @date 2023/3/1 * @description 历史数据的Excel工具类 */ public class HistoryExcelUtil { /** * 导出Excel * * @param sheetName sheet名称 * @param excelName 文件名 * @param heard 表头 * @param fields 对象属性 * @param dataList 要导出的数据 * @param response 输出流对象 * @param dataChartCostStr base64图片 */ public static <T> Mono<Void> getHSSFWorkbook(String sheetName, String excelName, String[] heard, String[] fields , List<T> dataList, ServerHttpResponse response, String dataChartCostStr) throws Exception { HSSFWorkbook workBook = new HSSFWorkbook(); HSSFSheet sheet = workBook.createSheet(Strings.isNullOrEmpty(sheetName) ? "场地历史数据" : sheetName); sheet.setDefaultColumnWidth(15); //设置宽度高度 sheet = setHeightAndWidth(sheet, heard); //创建单元格样式 HSSFCellStyle headStyle1 = makeCellStyle(workBook); //创建绘图(画布),注明:一个sheet只能创建一个画布,但一个画布中可以添加多张图片 HSSFPatriarch patriarch = sheet.createDrawingPatriarch(); //构成-图表图片 createPictureInExcel(dataChartCostStr, patriarch, workBook, (short) 1, dataList.size() + 5, (short) (heard.length+8), dataList.size() + 25); int index = 0; if (Objects.nonNull(heard)) { HSSFRow row = sheet.createRow(index); for (int i = 0; i < heard.length; i++) { HSSFCell cell = row.createCell(i); cell.setCellValue(heard[i]); cell.setCellStyle(headStyle1); } } if (CollectionUtils.isNotEmpty(dataList)) { for (int i = 0; i < dataList.size(); i++) { HSSFRow row = sheet.createRow(i + 1); for (int j = 0; j < fields.length; j++) { HSSFCell cell = row.createCell(j); String fieldName = fields[j]; String getMethodName = "get" + fieldName.substring(0, 1).toUpperCase() + fieldName.substring(1); T t = (T) dataList.get(i); Class tCls = t.getClass(); System.out.println(getMethodName); Method getMethod = tCls.getMethod(getMethodName, new Class[]{}); Object value = getMethod.invoke(t, new Object[]{}); // 判断值的类型后进行强制类型转换 String textValue = null; if (value == null) { cell.setCellValue(""); } if (value instanceof Integer) { int intValue = (Integer) value; cell.setCellValue(intValue); } else if (value instanceof Float) { float fValue = (Float) value; cell.setCellValue(fValue); } else if (value instanceof Double) { double dValue = (Double) value; cell.setCellValue(dValue); } else if (value instanceof Long) { long longValue = (Long) value; cell.setCellValue(longValue); } else if (value instanceof Date) { Date date = (Date) value; SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); textValue = sdf.format(date); cell.setCellValue(textValue); } else { // 其它数据类型都当作字符串简单处理 textValue = value == null ? "" : value.toString(); cell.setCellValue(textValue); } } } } return downloadExcelFile(response, workBook, excelName); } /** * 将图片保存到excel中 * * @param dataChart 图片的BASE64格式编码 * @param patriarch Excel-sheet 画布 * @param wb Excel工作簿 * @param col1 指定起始的单元格行索引 * @param row1 指定起始的单元格列索引 * @param col2 指定结束的单元格行索引 * @param row2 指定结束的单元格列索引 * @throws Exception */ @SuppressWarnings("restriction") public static void createPictureInExcel(String dataChart, HSSFPatriarch patriarch, HSSFWorkbook wb, short col1, int row1, short col2, int row2) throws Exception { //用于将BASE64编码格式转为byte数组 BASE64Decoder base64Decoder = new BASE64Decoder(); ByteArrayOutputStream dataChartoutStream = new ByteArrayOutputStream(); //将dataChartStringin作为输入流,读取图片存入image中 ByteArrayInputStream dataChartin = new ByteArrayInputStream(base64Decoder.decodeBuffer(dataChart)); BufferedImage dataChartbufferImg = ImageIO.read(dataChartin); //利用HSSFPatriarch将图片写入EXCEL ImageIO.write(dataChartbufferImg, "png", dataChartoutStream); /* * 指定绘图区域位置及大小 * HSSFClientAnchor(int dx1, int dy1, int dx2, int dy2, short col1, int row1, short col2, int row2) * 参数说明: * dx1 dy1 起始单元格中的x,y坐标. * dx2 dy2 结束单元格中的x,y坐标. * col1,row1 指定起始的单元格,下标从0开始. * col2,row2 指定结束的单元格 ,下标从0开始. * 详情参考博客 https://www.cnblogs.com/1175429393wljblog/p/9809868.html */ HSSFClientAnchor anchorCostStr = new HSSFClientAnchor(0, 0, 0, 0, col1, row1, col2, row2); //画图 patriarch.createPicture(anchorCostStr, wb.addPicture(dataChartoutStream.toByteArray(), HSSFWorkbook.PICTURE_TYPE_PNG)); } private static HSSFSheet setHeightAndWidth(HSSFSheet sheet, String[] heads) { //默认高度 sheet.setDefaultRowHeight((short) (20 * 30)); //控制第0列的宽度 sheet.setColumnWidth(0, 40 * 200); for (int i = 1; i < heads.length; i++) { sheet.setColumnWidth(i, 30 * 200); } return sheet; } /** * 设置表格风格 * * @param wb 工作簿对象 * @return */ private static HSSFCellStyle makeCellStyle(HSSFWorkbook wb) { //设置表格样式 HSSFCellStyle cellStyle = wb.createCellStyle(); //自动换行 cellStyle.setWrapText(true); //水平居中 cellStyle.setAlignment(CellStyle.ALIGN_CENTER); //垂直居中 cellStyle.setVerticalAlignment(CellStyle.VERTICAL_CENTER); //设置字体 HSSFFont font = wb.createFont(); font.setFontName("宋体"); //设置字体粗细 font.setBoldweight(Font.DEFAULT_CHARSET); //设置字体大小 font.setFontHeightInPoints((short) 13); font.setColor(Font.COLOR_RED); cellStyle.setFont(font); //设置边框 cellStyle.setBorderBottom(CellStyle.BORDER_MEDIUM); cellStyle.setBorderRight(CellStyle.BORDER_MEDIUM); cellStyle.setBorderLeft(CellStyle.BORDER_MEDIUM); cellStyle.setBorderTop(CellStyle.BORDER_MEDIUM); cellStyle.setFillForegroundColor(HSSFColor.BLUE.index2); cellStyle.setFillBackgroundColor(HSSFColor.BLUE.index2); cellStyle.setBottomBorderColor(HSSFColor.DARK_RED.index2); return cellStyle; } public static Mono<Void> downloadExcelFile(ServerHttpResponse response, HSSFWorkbook hssfWorkbook, String exclName) throws IOException { ZeroCopyHttpOutputMessage response1 = (ZeroCopyHttpOutputMessage) response; HttpHeaders headers = response1.getHeaders(); // 设置被下载的文件名称 headers.set(HttpHeaders.CONTENT_DISPOSITION, "attachment; filename=" + URLEncoder.encode(exclName, "UTF-8")); headers.add("Accept-Ranges", "bytes"); headers.set("file-name", URLEncoder.encode(exclName, "UTF-8")); headers.set("Access-Control-Allow-Origin", "*"); MediaType application = new MediaType("application", "octet-stream", StandardCharsets.UTF_8); headers.setContentType(application); // 定义输出流 DefaultDataBuffer dataBuffer = new DefaultDataBufferFactory().allocateBuffer(); OutputStream outputStream = dataBuffer.asOutputStream(); // 将excel文件流写入到output流 hssfWorkbook.write(outputStream); Flux<DataBuffer> dataBufferFlux = Flux.create((FluxSink<DataBuffer> emitter) -> { emitter.next(dataBuffer); emitter.complete(); }); return response1.writeAndFlushWith(Mono.just(dataBufferFlux)); } }
上一篇: java base64转图片
下一篇: 看过一个小故事