编程语言
首页 > 编程语言> > Java使用poi导出Excel之格式设置

Java使用poi导出Excel之格式设置

作者:互联网

最近接到一个需求,客户不满意原本导出的csv文件,想要导出Excel文件。不就导出Excel文件嘛,小意思,于是乎信心满满从网上扒导出的代码,一顿CV大法,搞定!代码如下:

import lombok.extern.slf4j.Slf4j;
import org.apache.poi.ss.usermodel.BorderStyle;
import org.apache.poi.ss.usermodel.HorizontalAlignment;
import org.apache.poi.xssf.usermodel.*;

import java.io.File;
import java.io.FileOutputStream;
import java.math.BigDecimal;
import java.text.SimpleDateFormat;
import java.util.*;

@Slf4j
public class ExcelUtils {

    // 新版Excel文件后缀
    private static final String EXCEL_SUFFIX = ".xlsx";

    /**
     * 导出核心实现
     *
     * @param fileName
     * @param headers
     * @param dataList
     * @return XSSFWorkbook
     */
    @Deprecated
    private static XSSFWorkbook writeDataToWorkbook(String fileName, List<String> headers, List<Map<String, Object>> dataList) {
        // 创建一个工作薄
        XSSFWorkbook workbook = new XSSFWorkbook();
        try {
            // 创建一个工作表
            XSSFSheet sheet = workbook.createSheet(fileName);

            // 设置表头字体格式
            XSSFFont headersFont = workbook.createFont();
            headersFont.setColor(new XSSFColor(java.awt.Color.DARK_GRAY));
            headersFont.setFontHeightInPoints((short) 14);
            headersFont.setBold(true);

            // 设置正文字体格式
            XSSFFont dataSetFont = workbook.createFont();
            dataSetFont.setColor(new XSSFColor(java.awt.Color.BLACK));
            dataSetFont.setBold(false);

            // 创建表头样式
            XSSFCellStyle headersStyle = workbook.createCellStyle();
            headersStyle.setBorderTop(BorderStyle.THIN);
            headersStyle.setBorderBottom(BorderStyle.THIN);
            headersStyle.setBorderLeft(BorderStyle.THIN);
            headersStyle.setBorderRight(BorderStyle.THIN);
            headersStyle.setFont(headersFont);
            // 表头内容对齐方式:居中
            headersStyle.setAlignment(HorizontalAlignment.CENTER);

            // 创建文本样式
            XSSFCellStyle textStyle = workbook.createCellStyle();
            textStyle.setBorderBottom(BorderStyle.THIN);
            textStyle.setBorderRight(BorderStyle.THIN);
            textStyle.setBorderLeft(BorderStyle.THIN);
            textStyle.setFont(dataSetFont);
            // 数据内容对齐方式:居左
            textStyle.setAlignment(HorizontalAlignment.LEFT);

            // 创建数字样式
            XSSFCellStyle numeralStyle = workbook.createCellStyle();
            numeralStyle.setBorderBottom(BorderStyle.THIN);
            numeralStyle.setBorderRight(BorderStyle.THIN);
            numeralStyle.setBorderLeft(BorderStyle.THIN);
            numeralStyle.setFont(dataSetFont);
            // 数据内容对齐方式:居右
            numeralStyle.setAlignment(HorizontalAlignment.RIGHT);

            // 此处设置数据格式
            XSSFDataFormat df = workbook.createDataFormat();

            int index = 0;
            // 创建表头并设置样式
            XSSFRow row = sheet.createRow(index);
            for (int i = 0; i < headers.size(); i++) {
                sheet.setColumnWidth(i, 20 * 256);
                XSSFCell cell = row.createCell(i);
                cell.setCellStyle(headersStyle);
                XSSFRichTextString text = new XSSFRichTextString(headers.get(i));
                cell.setCellValue(text);
            }
            // 导出正文数据,并设置其样式
            for (Map<String, Object> data : dataList) {
                index++;
                row = sheet.createRow(index);
                int column = 0;
                for (String key : data.keySet()) {
                    XSSFCell cell = row.createCell(column++);
                    Object value = data.get(key);
                    if (value == null) {
                        continue;
                    }

                    String dataType = value.getClass().getName();
                    if (dataType.endsWith("BigDecimal") || dataType.endsWith("Double")) {
                        // 带小数点的数字格式
                        cell.setCellStyle(numeralStyle);
                        numeralStyle.setDataFormat(df.getFormat("0.00"));//保留两位小数点
                        cell.setCellValue(Double.parseDouble(value.toString()));
                    } else if (dataType.endsWith("Integer") || dataType.endsWith("Long")) {
                        // 整型数字格式
                        cell.setCellStyle(numeralStyle);
                        numeralStyle.setDataFormat(df.getFormat("General"));//数据格式采用常规
                        cell.setCellValue(Double.parseDouble(value.toString()));
                    } else if (dataType.endsWith("Date")) {
                        //日期转为字符串
                        cell.setCellStyle(textStyle);
                        cell.setCellValue(date2Str((Date) value, "yyyy-MM-dd HH:mm:ss"));
                    } else {
                        // 文本格式
                        cell.setCellStyle(textStyle);
                        cell.setCellValue(value.toString());
                    }
                }
            }
        } catch (Exception e) {
            log.error("writeDataToWorkbook error, ", e);
        }
        return workbook;
    }

    /**
     * 测试类
     */
    public static void main(String[] args) {
        String filename = date2Str(new Date(), "yyyyMMddHHmmss");
        List<String> headers = Arrays.asList("客户编号", "手机号", "姓名", "创建时间", "账户余额", "账户余额1",  "账户余额2", "年龄");
        List<Map<String, Object>> datas = new ArrayList<>();
        Map<String, Object> map = new LinkedHashMap<>();
        map.put("id", 123);
        map.put("mobile", "0833344545");
        map.put("name", "jfdkdb ft#E@");
        map.put("time", new Date());
        map.put("amount", new BigDecimal("34276.8601"));
        map.put("amount1", 34276.8);
        map.put("amount2", 323455.10);
        map.put("days", 24);
        datas.add(map);
        XSSFWorkbook wb = writeDataToWorkbook(filename, headers, datas);
        try {
            File file = new File("/Users/test/" + filename + EXCEL_SUFFIX);
            FileOutputStream fileOutputStream = new FileOutputStream(file);
            wb.write(fileOutputStream);
            fileOutputStream.close();
        } catch (Exception e) {
            System.err.println(e.getMessage());
        }
        System.out.println("export excel " + filename + " succeed!");
    }

    public static String date2Str(Date date, String pattern) {
        if (null == date) return "";
        SimpleDateFormat sdf = new SimpleDateFormat(pattern);
        return sdf.format(date);
    }
}

项目跑起来本地测试一波看看效果:

好像没啥毛病,收。。。。球都么得,余额那几列的小数点位数好像不对啊!

回首撸代码,我设置了啊, 咋就没生效呢???

于是乎继续网上扒资料,各种说法都试一波:

1.如下图加celltype的设置,测试效果和第一次一样,失败;

 2.我又猜测是不是转double类型导致的,就改成如下图所示,赋值的时候用字符串格式:

        测试效果和第一次一样,不,比第一次还差,小数位数不对就算了,单元格左上角多了个绿色的小箭头,意味着客户不能直接框选看总额了,失败;

这么简单个功能,我居然卡这里了,心态。。崩了啊。。。。

3. 是你逼我的。要出绝招了,我自己处理数据,于是改成这样:

  跑起来,效果不错,小数位数都对了,但是单元格还是文本格式的。。。。

 4. 数据格式都对了,单元格格式嘛,简单,设置下入参就好了,So easy!马上改成下图:

 再次跑起来,结果如下所示:崩了。。。彻底崩。。了!

投降了。。我决定,文本格式就文本格式吧,数据格式对了就行,准备使用方案3了。

就在我要commit的时候,我突然看到结果4中的余额数据有","分割,我擦,这不是我设置的整型格式吗???我决定再试一把,我新建个数字格式专门给浮点型的数据用,代码又改成如下:


import lombok.extern.slf4j.Slf4j;
import org.apache.poi.ss.usermodel.BorderStyle;
import org.apache.poi.ss.usermodel.HorizontalAlignment;
import org.apache.poi.xssf.usermodel.*;

import java.io.File;
import java.io.FileOutputStream;
import java.math.BigDecimal;
import java.text.SimpleDateFormat;
import java.util.*;

@Slf4j
public class ExcelUtils {

    // 新版Excel文件后缀
    private static final String EXCEL_SUFFIX = ".xlsx";

    /**
     * 导出核心实现
     *
     * @param fileName
     * @param headers
     * @param dataList
     * @return XSSFWorkbook
     */
    @Deprecated
    private static XSSFWorkbook writeDataToWorkbook(String fileName, List<String> headers, List<Map<String, Object>> dataList) {
        // 创建一个工作薄
        XSSFWorkbook workbook = new XSSFWorkbook();
        try {
            // 创建一个工作表
            XSSFSheet sheet = workbook.createSheet(fileName);

            // 设置表头字体格式
            XSSFFont headersFont = workbook.createFont();
            headersFont.setColor(new XSSFColor(java.awt.Color.DARK_GRAY));
            headersFont.setFontHeightInPoints((short) 14);
            headersFont.setBold(true);

            // 设置正文字体格式
            XSSFFont dataSetFont = workbook.createFont();
            dataSetFont.setColor(new XSSFColor(java.awt.Color.BLACK));
            dataSetFont.setBold(false);

            // 此处设置数据格式
            XSSFDataFormat df = workbook.createDataFormat();

            // 创建表头样式
            XSSFCellStyle headersStyle = workbook.createCellStyle();
            headersStyle.setBorderTop(BorderStyle.THIN);
            headersStyle.setBorderBottom(BorderStyle.THIN);
            headersStyle.setBorderLeft(BorderStyle.THIN);
            headersStyle.setBorderRight(BorderStyle.THIN);
            headersStyle.setFont(headersFont);
            // 表头内容对齐方式:居中
            headersStyle.setAlignment(HorizontalAlignment.CENTER);

            // 创建文本样式
            XSSFCellStyle textStyle = workbook.createCellStyle();
            textStyle.setBorderBottom(BorderStyle.THIN);
            textStyle.setBorderRight(BorderStyle.THIN);
            textStyle.setBorderLeft(BorderStyle.THIN);
            textStyle.setFont(dataSetFont);
            // 数据内容对齐方式:居左
            textStyle.setAlignment(HorizontalAlignment.LEFT);

            // 创建浮点型数字样式
            XSSFCellStyle floatStyle = workbook.createCellStyle();
            floatStyle.setBorderBottom(BorderStyle.THIN);
            floatStyle.setBorderRight(BorderStyle.THIN);
            floatStyle.setBorderLeft(BorderStyle.THIN);
            floatStyle.setAlignment(HorizontalAlignment.RIGHT);
            floatStyle.setFont(dataSetFont);
            floatStyle.setDataFormat(df.getFormat("#,##0.00"));

            // 创建整型数字样式
            XSSFCellStyle integerStyle = workbook.createCellStyle();
            integerStyle.setBorderBottom(BorderStyle.THIN);
            integerStyle.setBorderRight(BorderStyle.THIN);
            integerStyle.setBorderLeft(BorderStyle.THIN);
            integerStyle.setAlignment(HorizontalAlignment.RIGHT);
            integerStyle.setFont(dataSetFont);
            integerStyle.setDataFormat(df.getFormat("0"));

            int index = 0;
            // 创建表头并设置样式
            XSSFRow row = sheet.createRow(index);
            for (int i = 0; i < headers.size(); i++) {
                sheet.setColumnWidth(i, 20 * 256);
                XSSFCell cell = row.createCell(i);
                cell.setCellStyle(headersStyle);
                XSSFRichTextString text = new XSSFRichTextString(headers.get(i));
                cell.setCellValue(text);
            }
            // 导出正文数据,并设置其样式
            for (Map<String, Object> data : dataList) {
                index++;
                row = sheet.createRow(index);
                int column = 0;
                for (String key : data.keySet()) {
                    XSSFCell cell = row.createCell(column++);
                    Object value = data.get(key);
                    if (value == null) {
                        continue;
                    }

                    String dataType = value.getClass().getName();
                    if (dataType.endsWith("BigDecimal") || dataType.endsWith("Double") || dataType.endsWith("Float")) {
                        cell.setCellStyle(floatStyle);  // 带小数点的数字格式
                        cell.setCellValue(Double.parseDouble(value.toString()));
                    } else if (dataType.endsWith("Integer") || dataType.endsWith("Long")) {
                        cell.setCellStyle(integerStyle);    // 整型数字格式
                        cell.setCellValue(Double.parseDouble(value.toString()));
                    } else if (dataType.endsWith("Date")) {
                        cell.setCellStyle(textStyle);   //日期转为字符串
                        cell.setCellValue(date2Str((Date) value, "yyyy-MM-dd HH:mm:ss"));
                    } else {
                        cell.setCellStyle(textStyle);   // 文本格式
                        cell.setCellValue(value.toString());
                    }
                }
            }
        } catch (Exception e) {
            log.error("writeDataToWorkbook error, ", e);
        }
        return workbook;
    }

    /**
     * 测试类
     */
    public static void main(String[] args) {
        String filename = date2Str(new Date(), "yyyyMMddHHmmss");
        List<String> headers = Arrays.asList("客户编号", "手机号", "姓名", "创建时间", "账户余额", "账户余额1",  "账户余额2", "年龄");
        List<Map<String, Object>> datas = new ArrayList<>();
        Map<String, Object> map = new LinkedHashMap<>();
        map.put("id", 123);
        map.put("mobile", "0833344545");
        map.put("name", "jfdkdb ft#E@");
        map.put("time", new Date());
        map.put("amount", new BigDecimal("34276.8601"));
        map.put("amount1", 34276.8);
        map.put("amount2", 323455.10);
        map.put("days", 24);
        datas.add(map);
        XSSFWorkbook wb = writeDataToWorkbook(filename, headers, datas);
        try {
            File file = new File("/Users/test/" + filename + EXCEL_SUFFIX);
            FileOutputStream fileOutputStream = new FileOutputStream(file);
            wb.write(fileOutputStream);
            fileOutputStream.close();
        } catch (Exception e) {
            System.err.println(e.getMessage());
        }
        System.out.println("export excel succeed!");
    }

    public static String date2Str(Date date, String pattern) {
        if (null == date) return "";
        SimpleDateFormat sdf = new SimpleDateFormat(pattern);
        return sdf.format(date);
    }
}

结果如下:

我擦嘞,真的就好了???!

原来是

XSSFCellStyle integerStyle = workbook.createCellStyle();

创建出来的style只能赋值一次,后面的赋值无效导致的。

虽然被这个小问题折磨了快一天了,心态都崩了好几次,但是好在最后找到原因了,解决问题的感觉真爽,崩的稀碎的心态有重新凝聚起来,比以前更强大了呢。。。

标签:map,Java,Excel,cell,THIN,poi,workbook,new,BorderStyle
来源: https://blog.csdn.net/chengsi101/article/details/118731836