easyExcel类(Excel解析工具)

一、 介绍

Java解析、生成Excel比较有名的框架有Apache poi、jxl。但他们都存在一个严重的问题就是非常的耗内存,poi有一套SAX模式的API可以一定程度的解决一些内存溢出的问题,但POI还是有一些缺陷,比如07版Excel解压缩以及解压后存储都是在内存中完成的,内存消耗依然很大。easyexcel重写了poi对07版Excel的解析,一个3M的excel用POI sax解析依然需要100M左右内存,改用easyexcel可以降低到几M,并且再大的excel也不会出现内存溢出;03版依赖POI的sax模式,在上层做了模型转换的封装,让使用者更加简单方便

版本支持
2+版本支持Java7&Java6
3+版本支持Java8

详细了解可以点击官网文档查阅

个人依赖:

<dependencies>
        <!--easyexcel-->
        <dependency>
            <groupId>com.alibaba</groupId>
            <artifactId>easyexcel</artifactId>
            <version>2.1.6</version>
        </dependency>
        <!--slf4j 日志-->
        <dependency>
            <groupId>org.slf4j</groupId>
            <artifactId>slf4j-log4j12</artifactId>
            <version>1.7.30</version>
        </dependency>

</dependencies>

二、 读Excel

Excel:


对象:

@Data
public class Question1 {
    private String id;            //题目ID
    private String companyId;   //所属企业
    private String catalogId;   //题目所属目录ID
}

监视器:

名称 说明
AnalysisEventListener<T> 分析事件侦听器:接收解析的每条数据的返回
SyncReadListener 同步读取侦听器
AbstractIgnoreExceptionReadListener 抽象忽略异常读取侦听器
ModelBuildEventListener 模型构建事件侦听器

AnalysisEventListener<T>
所有已实现的接口:Listener、ReadListener
直接已知子类:SyncReadListener

方法 返回值类型 说明
boolean hasNext 验证是否有另一条数据。您可以通过返回 false 来停止读取
void invokeHead 分析第一行时触发调用函数
void invokeHeadMap 以map的形式放回表头,覆盖当前方法以接收表头数据
void onException 当任何一个监听器进行错误报告时,所有监听器都会收到此方法

代码:

 private static void read2() {
        final List list = new ArrayList();
        //使用EasyExcel读取test1.xlsx文件 
        EasyExcel.read("../test/test1.xlsx", Question1.class, new AnalysisEventListener<Question1>() {
                    //重写子类方法
                    @Override
                    public void invoke(Question1 question1, AnalysisContext analysisContext) {
                        list.add(question1);
                    }
                    //重写子类方法
                    @Override
                    public void doAfterAllAnalysed(AnalysisContext analysisContext) {

                    }
          
                    @Override
                    public void invokeHeadMap(Map headMap, AnalysisContext context) {
                        System.out.println(headMap);
                    }
                }
        ).doReadAll();

        //获取读取到的数据
        for (Object o : list) {
            Question1 question1 = (Question1) o;
            System.out.println(question1);
        }

    }

结果视图:


SyncReadListener
所有已实现的接口:Listener、ReadListener
直接已知父类:AnalysisEventListener<T>

方法 返回值类型 说明
void doAfterAllAnalysed 如果有什么操作在全部分析结束后执行
List<Object> getList
void invoke 当分析一行触发器调用函数
void setList(List<Object> list)

代码:

private static void read1() {
        final List list = new ArrayList();
        //使用EasyExcel读取test1.xlsx文件
        EasyExcel.read("../test/test1.xlsx", Question1.class, new SyncReadListener() {
            //EasyExcel在读取excel表格时,每读取到一行,就会调用一次这个方法,
            //并且将读取到的行数据,封装到指定类型(Question1)的对象中,传递给我们(Object object)
            /*
            此问题可能出现在低版本的easyExcel中,出现时可以按照下列方式解决
                如果表格数据不是顶行写的,需要通过headRowNumber指定表头行的数量
                如果表格数据不是顶列写的,需要在封装的实体属性上通过@ExcelProperty将实体属性和表格列名进行对应
             */
            @Override
            public void invoke(Object object, AnalysisContext context) {
//                System.out.println(object);
                list.add(object);
            }
        }).doReadAll();

        //获取读取到的数据
        for (Object o : list) {
            Question1 question1 = (Question1) o;
            System.out.println(question1);
        }

    }

解决方式

    //index属性指定当前这个对应的是表格中哪个索引的列,表格中的索引是从0开始的
    /**
     * 强制读取第三个 这里不建议 index 和 name 同时用,要么一个对象只用index,要么一个对象只用name去匹配
     */
    @ExcelProperty(index = 2)
    private String catalogId;   //题目所属目录ID
    /**
     * 用名字去匹配,这里需要注意,如果名字重复,会导致只有一个字段读取到数据
     */
    @ExcelProperty("题目id")
    private String id;      //题目ID
    @ExcelProperty("企业id")
    private String companyId;   //所属企业

三、 写Excel (比较常用)

3.1、简单入门


    private static void write() {
        List list = new ArrayList();
        list.add(new Question1("1", "1", "1"));
        list.add(new Question1("2", "1", "1"));
        list.add(new Question1("3", "1", "1"));
  1.
        EasyExcel
                .write("test/test_w.xlsx", Question1.class)
                .sheet()
                .doWrite(list);
    

  2.
    ExcelWriter excelWriter = null;
        try {
            excelWriter = EasyExcel.write("test/test_w.xlsx", Question1.class).build();
            WriteSheet writeSheet = EasyExcel.writerSheet("模板").build();
            excelWriter.write(list, writeSheet);
        } finally {
            // 千万别忘记finish 会帮忙关闭流
            if (excelWriter != null) {
                excelWriter.finish();
            }
        }
    }

3.2、 指定写入的列
添加该实体列的注释就可以实现如下:

    @ExcelProperty(value = "题目id", index = 0)
    private String id;      //题目ID
    @ExcelProperty(value = "企业id", index = 1)
    private String companyId;   //所属企业
    @ExcelProperty(value = "类目id", index = 3)
    private String catalogId;   //题目所属目录ID
指定写入的列

3.2、 复杂表头
添加该实体列的注释就可以实现如下:

    @ExcelProperty({"主标题", "题目id"})
    private String id;      //题目ID
    @ExcelProperty({"主标题", "企业id"})
    private String companyId;   //所属企业
    @ExcelProperty({"主标题", "类目id"})
    private String catalogId;   //题目所属目录ID
复杂表头

3.3、 列宽 行高
添加该实体列的注释就可以实现如下:

  @Data
  @HeadRowHeight(30) //表头行高
  @ContentRowHeight(20) //数据行高
  @ColumnWidth(25) //列宽
  public class Question1 {

    @ExcelProperty({"题目id"})
    private String id;      //题目ID
    @ExcelProperty({"企业id"})
    private String companyId;   //所属企业
    //局部定义
    @ColumnWidth(50)
    @ExcelProperty({"类目id"})
    private String catalogId;   //题目所属目录ID
  }
  • 自定义列宽 行高

三、填充Excel

3.1、基于模板填充表格数据

  1. 写出多条记录
private static void write_template_multi() {
        List list = new ArrayList();
        list.add(new Question1("1", "1", "1"));
        list.add(new Question1("2", "1", "1"));
        list.add(new Question1("3", "1", "1"));

        EasyExcel
                .write("../test/test_w.xlsx", Question1.class)
                .withTemplate("../test/test_template.xlsx")
                .sheet()
                .doFill(list);
    }
  • 多条数据

2.写出一条记录

private static void write_template_one() {
        EasyExcel
                .write("test/test_w.xlsx", Question1.class)
                .withTemplate("test/test_template_one.xlsx")
                .sheet()
                .doFill(new Question1("1", "1", "1"));
    }
  • 一条数据

添加一条数据与添加多条数据的差别如下:
模板中参数中的前是否有.,若有则能多或单条条数据写入;否则只允许单条数据写入

  • 一条模板
  • 多条模板

3.1、复杂的填充

private static void write_template_multi_one() {
        List list = new ArrayList();
        list.add(new Question1("1", "1", "1"));
        list.add(new Question1("2", "1", "1"));
        list.add(new Question1("3", "1", "1"));

        ExcelWriterBuilder writerBuilder = EasyExcel
                .write("../test/test_w.xlsx", Question1.class)
                .withTemplate("../test/test_template_mul_one.xlsx");

        WriteSheet writeSheet = writerBuilder.sheet().build();
        ExcelWriter excelWriter = writerBuilder.build();

        //在填充玩多结果数据后,要强制换行,不然后续的单结果数据会发生覆盖现象
        FillConfig fillConfig = FillConfig.builder().forceNewRow(true).build();

        excelWriter.fill(list, fillConfig, writeSheet);

        Map map = new HashMap();
        map.put("total", 600);
        excelWriter.fill(map, writeSheet);

        //调用finish方法
        excelWriter.finish();
    }

四、常见API

4.1、关于常见类解析

类名 说明
EasyExcel 入口类,用于构建开始各种操作
ExcelReaderBuilder ExcelWriterBuilder 构建出一个 ReadWorkbook WriteWorkbook,可以理解成一个excel对象,一个excel只要构建一个
ExcelReaderSheetBuilder ExcelWriterSheetBuilder 构建出一个 ReadSheet WriteSheet对象,可以理解成excel里面的一页,每一页都要构建一个
ReadListener 在每一行读取完毕后都会调用ReadListener来处理数据
WriteHandler 在每一个操作包括创建单元格、创建表格等都会调用WriteHandler来处理数据

所有配置都是继承的,Workbook的配置会被Sheet继承,所以在用EasyExcel设置参数的时候,在EasyExcel...sheet()方法之前作用域是整个sheet,之后针对单个sheet

4.2、读
注解
ExcelProperty 指定当前字段对应excel中的那一列。可以根据名字或者Index去匹配。当然也可以不写,默认第一个字段就是index=0,以此类推。千万注意,要么全部不写,要么全部用index,要么全部用名字去匹配。千万别三个混着用,除非你非常了解源代码中三个混着用怎么去排序的。
ExcelIgnore 默认所有字段都会和excel去匹配,加了这个注解会忽略该字段
DateTimeFormat 日期转换,用String去接收excel日期格式的数据会调用这个注解。里面的value参照java.text.SimpleDateFormat
NumberFormat 数字转换,用String去接收excel数字格式的数据会调用这个注解。里面的value参照java.text.DecimalFormat
ExcelIgnoreUnannotated 默认不加ExcelProperty 的注解的都会参与读写,加了不会参与

4.2、写
注解
ExcelProperty index 指定写到第几列,默认根据成员变量排序。value指定写入的名称,默认成员变量的名字,多个value可以参照快速开始中的复杂头
ExcelIgnore 默认所有字段都会写入excel,这个注解会忽略这个字段
DateTimeFormat 日期转换,将Date写到excel会调用这个注解。里面的value参照java.text.SimpleDateFormat
NumberFormat 数字转换,用Number写excel会调用这个注解。里面的value参照java.text.DecimalFormat
ExcelIgnoreUnannotated 默认不加ExcelProperty 的注解的都会参与读写,加了不会参与

五、其它相关特殊用法

六、文档保护措施

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,236评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,867评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,715评论 0 340
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,899评论 1 278
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,895评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,733评论 1 283
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,085评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,722评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,025评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,696评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,816评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,447评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,057评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,009评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,254评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,204评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,561评论 2 343

推荐阅读更多精彩内容