使用SpringEL功能动态解析模板Excel来生成Excel

SpringEL可以参考:https://www.cnblogs.com/yangzhilong/p/11282560.html

本文基于POI 3.1.7,spring 5.1.7编写。

由于excel里的#是数字的关键字,所有我使用${}来替换了#字符。

核心帮助类:

package com.longge.util;

import java.io.InputStream;
import java.util.ArrayList;
import java.util.List;

import org.apache.poi.ss.usermodel.Cell;
import org.apache.poi.ss.usermodel.CellStyle;
import org.apache.poi.ss.usermodel.CellType;
import org.apache.poi.ss.usermodel.Row;
import org.apache.poi.ss.usermodel.Sheet;
import org.apache.poi.ss.usermodel.Workbook;
import org.apache.poi.ss.usermodel.WorkbookFactory;
import org.springframework.core.io.ClassPathResource;
import org.springframework.expression.EvaluationContext;
import org.springframework.expression.ExpressionParser;
import org.springframework.expression.spel.standard.SpelExpressionParser;
import org.springframework.expression.spel.support.StandardEvaluationContext;
import org.springframework.util.CollectionUtils;

/**
 * @author yangzhilong
 * @date 8/01/2019
 */
public class ExcelTemplateWriteUtils {
    
    /**
     * get excel template Workbook
     * @param templateFilePath  eg: static/excelTemplate/xx.xlsx
     * @return
     */
    public static Workbook getWorkbook(String templateFilePath) {
        try {
            InputStream is;
            if(templateFilePath.startsWith("/")) {
                is = ExcelTemplateWriteUtils.class.getResourceAsStream(templateFilePath);
            } else {
                is = new ClassPathResource(templateFilePath).getInputStream();
            }
            return WorkbookFactory.create(is);
        } catch (Exception e) {
            throw new RuntimeException(e);
        }
    }
    
    /**
     * @param sheet
     * @param templateRowIndex
     * @param list
     */
    public static void fillDataWithTemplate(Sheet sheet, int templateRowIndex, List<? extends Object> list) {
        int tIndex = templateRowIndex;
        Row templateRow = sheet.getRow(templateRowIndex);
        if(CollectionUtils.isEmpty(list)) {
            sheet.removeRow(templateRow);
        } else {
            List<Object> template = new ArrayList<>(templateRow.getPhysicalNumberOfCells());
            List<CellStyle> sytles = new ArrayList<>(templateRow.getPhysicalNumberOfCells());
            templateRow.forEach(cell -> {
                sytles.add(cell.getCellStyle());
                cell.getCellStyle();
                CellType type = cell.getCellTypeEnum();
                if(CellType.NUMERIC.equals(type)) {
                    template.add(cell.getNumericCellValue());
                } else if(CellType.STRING.equals(type)) {
                    String value = cell.getStringCellValue();
                    if(value.startsWith("${")) {
                        value = formatEl(value);
                    }
                    template.add(value);
                } else if(CellType.BLANK.equals(type)) {
                    template.add(null);
                } else {
                    throw new RuntimeException("can't support cell type:" + type.toString());
                }
            });
            ExpressionParser parser = new SpelExpressionParser();
            for(int i=0,len=list.size(); i<len; i++) {
                Object item = list.get(i);
                
                Row newRow = sheet.createRow(tIndex++);
                EvaluationContext context = new StandardEvaluationContext();
                context.setVariable("item", item);
                
                ForEachUtils.forEach(0, template, (index, value) -> {
                    Cell newCell = newRow.createCell(index);
                    newCell.setCellStyle(sytles.get(index));
                    if(null != value) {
                        if (value instanceof String) {
                            String v = (String)value;
                            if(v.startsWith("#")) {
                                v = parser.parseExpression(v).getValue(context, String.class);
                            }
                            newCell.setCellValue(v);
                        } else if(value instanceof Number) {
                            newCell.setCellValue(((Number)value).doubleValue());
                        }
                    }
                });
            }
        }
    }
    
    private static String formatEl(String el) {
        return el.replace("${", "#").replace("}", "");
    }
}

单元测试类:

package com.longge.util;

import static org.junit.Assert.assertEquals;

import java.io.File;
import java.io.FileOutputStream;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.UUID;

import org.apache.poi.ss.usermodel.Sheet;
import org.apache.poi.ss.usermodel.Workbook;
import org.junit.Test;

import com.longge.dto.CustomerDto;

/**
 * @author yangzhilong
 * @date 8/01/2019
 */
public class ExcelTemplateWriteUtilsTest {
    @Test
    public void test() throws Exception {
        String path = "util/template.xlsx";
        //String path = "/util/template.xlsx";
        Workbook workbook = ExcelTemplateWriteUtils.getWorkbook(path);
        Sheet sheet = workbook.getSheetAt(0);
        
        List<CustomerDto> list = new ArrayList<>(10);
        for(int i=0; i<10; i++) {
            CustomerDto dto = new CustomerDto();
            dto.setChineseName("name:" + i);
            dto.setCustomerCode("code:" + i);
            dto.setOwnerGroup("group:" + i);
            list.add(dto);
        }
        
        ExcelTemplateWriteUtils.fillDataWithTemplate(sheet, 1, list);
        
        assertEquals(sheet.getPhysicalNumberOfRows(), 11);
        
        workbook.write(new FileOutputStream(new File("C:\\Users\\RYan27\\Desktop\\test\\" + UUID.randomUUID().toString() + ".xlsx")));
        workbook.close();
    }
}

模板文件:

最终生成的文件实例:

 

posted @ 2019-08-01 15:57  自行车上的程序员  阅读(831)  评论(0编辑  收藏  举报