当前位置: 首页 > article >正文

EasyExcel日常使用总结

文章目录

    • 概要
    • 引入依赖
    • 常用操作方法
      • 折叠或隐藏列
      • 折叠或隐藏行
      • 单元格样式
      • 单行表头设置
      • 多行表头设置
      • 多个sheet写入
      • 自动列宽

概要

EasyExcel日常使用总结。

引入依赖

引入依赖

		<dependency>
			<groupId>com.alibaba</groupId>
			<artifactId>easyexcel</artifactId>
			<version>3.3.2</version>
		</dependency>

常用操作方法

折叠或隐藏列

  1. 新建对象用于存储隐藏列数据,后续直接根据此数据进行折叠隐藏。
@Data
public class ExcelHiddenCellVo {
    // 分组坐标起始
    private int startHidden;
    // 分组坐标结束
    private int endHidden;
    // 是否折叠
    private Boolean ifCollapse;
    // 是否隐藏
    private Boolean ifHidden;
}
  1. 计算折叠隐藏列数据(逻辑可根据数据自定义,折叠隐藏的,可以是固定的也可以是动态的)
    private List<ExcelHiddenCellVo> calculateHiddenColumn() {
            List<ExcelHiddenCellVo> excelHiddenCellVos = new ArrayList<>();
            ExcelHiddenCellVo excelHiddenCellVo = new ExcelHiddenCellVo();
            excelHiddenCellVo.setStartHidden(monthValue + 1);
            excelHiddenCellVo.setEndHidden(12);
            excelHiddenCellVo.setIfHidden(false);
            excelHiddenCellVo.setIfCollapse(true);
            excelHiddenCellVos.add(excelHiddenCellVo);
            return excelHiddenCellVos;
    }
  1. 定义RowWriteHandler类方法,在写入每一行数据之前或之后执行自定义操作
private RowWriteHandler rowWriteHandler(List<ExcelHiddenCellVo> excelHiddenCellVos) {
        return new RowWriteHandler() {
            @Override
            public void afterRowCreate(WriteSheetHolder writeSheetHolder, WriteTableHolder writeTableHolder, Row row, Integer relativeRowIndex, Boolean isHead) {
                // 列分组
                if (isHead && relativeRowIndex == 0) {
                    excelHiddenCellVos.forEach(item -> {
                        Sheet sheet = writeSheetHolder.getSheet();
                        sheet.groupColumn(item.getStartHidden(), item.getEndHidden());
                        sheet.setColumnGroupCollapsed(item.getStartHidden(), item.getIfCollapse());
                    });
                }
            }
        };
  }
  1. 使用的话直接注册一个写入处理器,把RowWriteHandler操作写入。
EasyExcel.write(outputStream)
                    .head(this.emailExcelHead(year, deptName)) // 设置表头
                    .inMemory(true)
                    .registerWriteHandler(this.rowWriteHandler(this.calculateHiddenColumn()))
                    .doWrite(deptKpiStatisticsExportVos);

折叠或隐藏行

和列是类似的,只是使用的Hanlder不一样

  1. 计算折叠隐藏行数据(逻辑可根据数据自定义,折叠隐藏的,可以是固定的也可以是动态的)
// 计算出需要折叠的行
    private List<ExcelHiddenCellVo> calculateHiddenRow(List<DeptKpiStatisticsExportVo> vos) {
        List<ExcelHiddenCellVo> excelHiddenCellVos = new ArrayList<>();
		......
  1. 定义CellWriteHandler类方法,在单元格创建后触发行折叠
private CellWriteHandler cellStyleHandler(List<ExcelHiddenCellVo> excelHiddenCellVos) {
        return new CellWriteHandler() {
            @Override
            public void afterCellCreate(WriteSheetHolder writeSheetHolder, WriteTableHolder writeTableHolder, Cell cell, Head head, Integer relativeRowIndex, Boolean isHead) {
                    excelHiddenCellVos.forEach(item -> {
                        Sheet sheet = writeSheetHolder.getSheet();
                        sheet.groupRow(item.getStartHidden(), item.getEndHidden());
                        sheet.setRowGroupCollapsed(item.getStartHidden(), true);
                    });
                }
            }
}            
  1. 使用的话直接注册一个写入处理器,把CellWriteHandler操作写入。
EasyExcel.write(outputStream)
                    .head(this.emailExcelHead(year, deptName)) // 设置表头
                    .inMemory(true)
                    .registerWriteHandler(this.rowWriteHandler(this.calculateHiddenColumn()))
                    .registerWriteHandler(this.cellStyleHandler(excelHiddenCellVos))
                    .doWrite(deptKpiStatisticsExportVos);

单元格样式

  1. 我一般使用自定义HorizontalCellStyleStrategy类,作为整体的默认样式
// 整体默认的样式
    private HorizontalCellStyleStrategy getHorizontalCellStyleStrategy() {
        WriteCellStyle headWriteCellStyle = new WriteCellStyle();
        WriteFont headWriteFont = new WriteFont();
        headWriteFont.setFontHeightInPoints((short) 9);
        headWriteFont.setBold(true);
        headWriteFont.setFontName("微软雅黑");
        headWriteCellStyle.setWriteFont(headWriteFont);
        // 内容的策略
        WriteCellStyle contentWriteCellStyle = new WriteCellStyle();
        // 这里需要指定 FillPatternType 为FillPatternType.SOLID_FOREGROUND 不然无法显示背景颜色.头默认了 FillPatternType所以可以不指定
        contentWriteCellStyle.setFillPatternType(FillPatternType.SOLID_FOREGROUND);
        // 背景绿色
        contentWriteCellStyle.setFillForegroundColor(IndexedColors.WHITE.getIndex());
        WriteFont contentWriteFont = new WriteFont();
        // 字体大小
        contentWriteFont.setFontHeightInPoints((short) 9);
        contentWriteFont.setFontName("微软雅黑");
        contentWriteCellStyle.setWriteFont(contentWriteFont);

        contentWriteCellStyle.setHorizontalAlignment(HorizontalAlignment.CENTER); // 设置水平居中
        contentWriteCellStyle.setBorderTop(BorderStyle.THIN); // 设置边框
        contentWriteCellStyle.setBorderBottom(BorderStyle.THIN);
        contentWriteCellStyle.setBorderLeft(BorderStyle.THIN);
        contentWriteCellStyle.setBorderRight(BorderStyle.THIN);
        // 这个策略是 头是头的样式 内容是内容的样式 其他的策略可以自己实现
        return new HorizontalCellStyleStrategy(headWriteCellStyle, contentWriteCellStyle);
    }
  1. 需要自定义样式的,再写一个CellWriteHandler类。
// 创建自定义的单元格样式
    private CellWriteHandler cellStyleHandler(List<FinanceDeptKpiStatisticsExportVo> financeDeptKpiStatisticsExportVos, List<ExcelHiddenCellVo> excelHiddenCellVos) {
	return new CellWriteHandler() {
			@Override
            public void afterCellDispose(CellWriteHandlerContext context) {
                Cell cell = context.getCell();
                WriteCellData<?> cellData = context.getFirstCellData();
                WriteCellStyle writeCellStyle = cellData.getOrCreateStyle();
				if (cell.getRowIndex() == 0 || cell.getRowIndex() == 1 || cell.getRowIndex() == 2) {
                    WriteFont writeFont = new WriteFont();
                    writeFont.setBold(true);
                    writeFont.setFontName("微软雅黑");
                    writeFont.setFontHeightInPoints((short) 14);

                    writeCellStyle.setBorderTop(BorderStyle.NONE); // 设置边框
                    writeCellStyle.setBorderBottom(BorderStyle.NONE);
                    writeCellStyle.setBorderLeft(BorderStyle.NONE);
                    writeCellStyle.setBorderRight(BorderStyle.NONE);
                    writeCellStyle.setFillPatternType(FillPatternType.SOLID_FOREGROUND);
                    writeCellStyle.setFillForegroundColor(IndexedColors.WHITE.getIndex());
                    writeCellStyle.setWriteFont(writeFont);
                }
          }      
	}
}
      
  1. 使用同样的注册到registerWriteHandler里面。

单行表头设置

  1. 单行表头可以定义一个类,使用@ExcelProperty注解标识每个字段。
@Data
public class ExportVo {
    @ExcelProperty(index = 0, value = "工号")
    private String headAccount;

    @ExcelProperty(index = 1, value = "姓名")
    private String headName;
}    
  1. 使用
EasyExcel.write(outputStream)
                    .head(ExportVo.class) // 设置表头

多行表头设置

使用List<List>嵌套,List里面元素有多少个,则表示有多少级表头。以下为四级表头。

// 表头设置
    private List<List<String>> emailExcelHead(String year, String deptName) {
        String title = deptName + "明细表";
        String titleYear = "(" + year + "年度)";
        List<List<String>> list = new ArrayList<>();
        List<String> head1 = new ArrayList<>();
        head1.add(title);
        head1.add(titleYear);
        head1.add("部门:");
        head1.add("序号");
        List<String> head2 = new ArrayList<>();
        head2.add(title);
        head2.add(titleYear);
        head2.add(deptName);
        head2.add("科目");
        List<String> head3 = new ArrayList<>();
        head3.add(title);
        head3.add(titleYear);
        head3.add("");
        head3.add("费用类型");
        list.add(head1);
        list.add(head2);
        list.add(head3);
        return list;

多个sheet写入

ByteArrayOutputStream outputStream = new ByteArrayOutputStream();
            ExcelWriter excelWriter = EasyExcel.write(outputStream).build();
            WriteSheet writeSheet = EasyExcel.writerSheet("报表1")
                    .head(CompleteWarehousingManHourExportVo.class)
                    .registerWriteHandler(new ExcelWidthStyleStrategy(0)) // 自定义列宽策略
                    .registerWriteHandler(WriteExcelUtils.getHorizontalCellStyleStrategy()) // 整体单元格的样式设置
                    .registerWriteHandler(this.CompleteWarehousingStyleHandler()) // 自定义单元格样式
                    .registerWriteHandler(this.rowWriteHandler(this.calculateHiddenColumn())) // 列分组
                    .build();
            excelWriter.write(warehousingManHourExportVos, writeSheet);

            WriteSheet writeSheet2 = EasyExcel.writerSheet("报表2")
                    .head(CompleteMesManHourExportVo.class)
                    .registerWriteHandler(new ExcelWidthStyleStrategy(0)) // 自定义列宽策略
                    .registerWriteHandler(WriteExcelUtils.getHorizontalCellStyleStrategy()) // 整体单元格的样式设置
                    .build();
            excelWriter.write(completeMesManHourExportVos, writeSheet2);
            excelWriter.finish();
            outputStream.flush();
            outputStream.close();
            InputStream inputStream = new ByteArrayInputStream(outputStream.toByteArray());
 

自动列宽

使用同样registerWriteHandler注册

public class ExcelWidthStyleStrategy extends AbstractColumnWidthStyleStrategy {

    // 单元格的最大宽度
    private static final int MAX_COLUMN_WIDTH = 8;
    // 缓存(第一个Map的键是sheet的index, 第二个Map的键是列的index, 值是数据长度)
    private  Map<Integer, Map<Integer, Integer>> CACHE = new HashMap(8);
    // 表头 0为第a行,1为第2行 依次类推
    Integer headRow = 0;

    public ExcelWidthStyleStrategy(Integer headRow){
        this.headRow = headRow;
    }

    // 重写设置列宽的方法
    @Override
    protected void setColumnWidth(WriteSheetHolder writeSheetHolder,
                                  List<WriteCellData<?>> cellDataList,
                                  Cell cell,
                                  Head head,
                                  Integer relativeRowIndex,
                                  Boolean isHead) {
        boolean needSetWidth = isHead || !CollectionUtils.isEmpty(cellDataList);
        // 当时表头或者单元格数据列表有数据时才进行处理
        if (needSetWidth) {
            Map<Integer, Integer> maxColumnWidthMap =
                    CACHE.get(writeSheetHolder.getSheetNo());

            if (maxColumnWidthMap == null) {
                maxColumnWidthMap = new HashMap(16);
                CACHE.put(writeSheetHolder.getSheetNo(), maxColumnWidthMap);
            }
            // 获取数据长度
            Integer columnWidth = this.getLength(cellDataList, cell, cell.getRowIndex());
            if (columnWidth >= 0) {
                if (columnWidth > MAX_COLUMN_WIDTH) {
                    columnWidth = MAX_COLUMN_WIDTH;
                }
                // 确保一个列的列宽以表头为主,如果表头已经设置了列宽,单元格将会跟随表头的列宽
                Integer maxColumnWidth = maxColumnWidthMap.get(cell.getColumnIndex());
                if (maxColumnWidth == null || columnWidth > maxColumnWidth) {
                    // 控制以哪一行为表头,大于1则以第二行
                    if (cell.getRowIndex() >= headRow)
                        maxColumnWidthMap.put(cell.getColumnIndex(), columnWidth);
                    // 如果使用EasyExcel默认表头,那么使用columnWidth * 512
                    // 如果不使用EasyExcel默认表头,那么使用columnWidth * 256
                    // 如果是自己定义的字体大小,可以再去测试这个参数常量
                    writeSheetHolder
                            .getSheet()
                            .setColumnWidth(cell.getColumnIndex(), columnWidth * 512);
                }

            }
        }
    }

/**
     * 获取当前单元格的数据长度
     * @param cellDataList
     * @param cell
     * @param
     * @return
     */
    private Integer getLength(List<WriteCellData<?>> cellDataList,
                              Cell cell,
                              Integer rowNum) {
        if (rowNum == headRow) {
            return cell.getStringCellValue().getBytes().length;
        } else {
            WriteCellData cellData = cellDataList.get(0);
            CellDataTypeEnum type = cellData.getType();
            if (type == null) {
                return -1;
            } else {
            switch(type) {
                    case STRING:
                        return cellData.getStringValue().getBytes().length;
                    case BOOLEAN:
                        return cellData.getBooleanValue().toString().getBytes().length;
                    case NUMBER:
                        return cellData.getNumberValue().toString().getBytes().length;
                    default:
                        return -1;
                }
            }
        }
    }

}

http://www.kler.cn/news/318801.html

相关文章:

  • 【C语言】使用 OpenSSL 进行 AES CBC 加密与解密
  • linux静态路由表
  • Nginx静态资源优化、压缩、缓存处理
  • 像百度谷歌这种网站会被DDoS吗
  • 全球商旅,提升人效,就选用友BIP超级版!
  • 【果蔬识别系统】Python+卷积神经网络算法+人工智能+深度学习+计算机毕设项目+Django网页界面平台
  • 基于springboot在线点餐系统
  • 2024源代码加密软件TOP10排行榜丨保护企业源代码安全不泄露
  • [数据结构与算法·C++版] 笔记 1.2 什么是数据结构
  • docker快速部署zabbix
  • 快递智能地址解析API接口代码
  • 支持K歌音箱方案应用的高性能 32 位蓝牙音频应用处理器-BP1048B2
  • 【解决方案】git错误:对象文件为空 error: object file
  • 考研日语 - 高频核心 2200 词(五)
  • 【算法】模拟:(leetcode)495.提莫攻击(easy)
  • linux服务器自动同步数据库
  • vue3腾讯云实时音视频通话 ui集成方案TUIcallkit
  • 富格林:可信交易筹划在线曝光
  • Python 解析 html
  • 股指期货理论价格计算公式是什么?
  • 【论文】FunAudioLLM:一个旨在增强人类与大型语言模型(LLMs)之间自然语音交互的模型家族
  • 如何在算家云搭建DiffSynth-Kolors-Painter(图像生成)
  • 【ARM】AMBA和总线
  • 详解机器学习经典模型(原理及应用)——随机森林
  • 【JS】严格模式/非严格模式的区别
  • Android 平台使用VIA创建语音交互应用开发指南
  • 更新C语言题目
  • 【linux】gcc makefile
  • 从0到1搭建权限管理系统系列三 .net8 JWT创建Token并使用
  • IDEA中Quarkus框架(3.13版本)开发、调试、部署、打包等