【webFlux Excel导出插入图片】POI导出Excel的时候插入图片

  • 作者: 凯哥Java(公众号:凯哥Java)
  • 工作小总结
  • 时间:2023-03-05 14:08
  • 16205人已阅读
简介 在工作中,有时候,我们会遇到这样的需求:导出数据的时候,同时把图片导出。本文使用于WebFlux响应式开发导出Excel功能。SpringMVC模式可以在此基础上做少许修改即可实现在SpringMVC模式导出带有图片的需求如下:导出效果:代码如下:入口:public Mono<Void> exportHistoryReport(HistoryDataParam&nb

🔔🔔🔔好消息!好消息!🔔🔔🔔

有需要的朋友👉:联系凯哥 微信号 kaigejava2022

在工作中,有时候,我们会遇到这样的需求:导出数据的时候,同时把图片导出。

本文使用于WebFlux 响应式开发导出Excel功能。SpringMVC模式可以在此基础上做少许修改即可实现在SpringMVC模式导出带有图片的

需求如下:

852e7c5a71a9659ae4f5e5ffb41f0622.png

导出效果:

f59d5b08135eed2a21db9caf035a240c.png

代码如下:

入口:

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转图片

下一篇: 看过一个小故事

TopTop