数据导入导出(EasyExcel)框架入门指南

news2025/1/15 16:55:13

写在前面
前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站


文章目录

  • EasyExcel 框架
    • 概述
    • 依赖
    • API
      • Excel 实体类注解
      • 写 Excel
        • 概念介绍
        • 写 Excel 通用参数
        • WriteWorkbook
        • WriteSheet
        • WriteTable
  • 代码示例
    • 创建 Excel 文件
      • (1)配置实体类
      • (2)Excel 文件的生成和下载
    • 模板打印
      • (1)定义模板
      • (2)填充模板并下载 Excel 模板文件
    • 解析 Excel 文件
      • (1)配置实体类
      • (2)解析 Excel 文件
  • 拓展
    • csv 读写的支持

EasyExcel 框架

本文仅作为快速入门指南,深入学习详见官方文档

概述

Java 解析、生成 Excel 比较有名的框架有Apache poi、jxl,但它们都存在一些严重的问题,如代码编写繁琐且重复,极其耗费内存。EasyExcel 是一个基于 Java 的简单、省内存的读写 Excel 的开源项目。在尽可能节约内存的情况下支持读写百万级数据的 Excel。

EasyExcel:阿里巴巴开源的操作 Excel 的框架,底层封装的 POI,可以大大减少占用内存的主要原因是在解析 Excel 时,没有将文件数据一次性全部加载到内存中,而是从磁盘上一行行读取数据,逐个解析。

下图是 easyExcel 和 POI 在解析 Excel 时的对比图:

在这里插入图片描述


easyExcel 采用一行一行的解析模式,并将一行的解析结果以观察者的模式通知处理(AnalysisEventListener)。

在这里插入图片描述

  • 在实体类上配置注解(适配 Excel 中的标题)
  • 调用 EasyExcel 的工具类方法,完成文件的上传或者下载

依赖

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

API

Excel 实体类注解

  • @ExcelIgnoreUnannotated:标注在实体类上,作用是只使用标注了 @ExcelProperty 注解的字段参与 excel 读写。默认是不管加不加 ExcelProperty 的注解,所有字段都会参与读写

  • @ExcelProperty:标注在实体类属性上,用于 excel 的列和实体类属性的匹配

    参数如下:

    名称默认值描述
    value用于匹配excel中的头,必须全匹配,如果有多行头,会匹配最后一行头
    orderInteger.MAX_VALUE优先级高于value,会根据order的顺序来匹配实体和 excel 中数据的顺序
    index-1优先级高于valueorder,会根据index直接指定到excel中具体的哪一列
    converter自动选择指定当前字段用什么转换器,默认会自动选择。写的情况下只要实现com.alibaba.excel.converters.Converter#convertToExcelData(com.alibaba.excel.converters.WriteConverterContext<T>) 方法即可
  • @ExcelIgnore:标注在实体类属性上,默认所有字段都会和 excel 去匹配,加了这个注解会忽略该字段

  • @DateTimeFormat:标注在实体类属性上,日期转换

    参数如下:

    名称默认值描述
    value参照java.text.SimpleDateFormat书写即可
    use1904windowing自动选择excel中时间是存储1900年起的一个双精度浮点数,但是有时候默认开始日期是1904,所以设置这个值改成默认1904年开始
  • @NumberFormat:标注在实体类属性上,数字转换

    参数如下:

    名称默认值描述
    value参照java.text.DecimalFormat书写即可
    roundingModeRoundingMode.HALF_UP格式化的时候设置舍入模式

写 Excel

概念介绍
  • WriteWorkbook :可以理解成一个 excel 文件
  • WriteSheet :理解成一个excel 文件里面的一个表单
  • WriteTable : 一个表单里面如果有多个实际用的表格,则可以用 WriteTablejava

写 Excel 通用参数

WriteWorkbook,WriteSheet ,WriteTable 都会有的参数,如果为空,默认使用上级。

名称默认值描述
converter默认加载了很多转换器,这里可以加入不支持的字段
writeHandler写的处理器。可以实现WorkbookWriteHandler,SheetWriteHandler,RowWriteHandler,CellWriteHandler,在写入excel的不同阶段会调用
relativeHeadRowIndex0写入到excel和上面空开几行
headclazz二选一。读取文件头对应的列表,会根据列表匹配数据,建议使用class
clazzhead二选一。读取文件的头对应的class,也可以使用注解。如果两个都不指定,则会读取全部数据
autoTrimtrue会对头、读取数据等进行自动trim
use1904windowingfalseexcel中时间是存储1900年起的一个双精度浮点数,但是有时候默认开始日期是1904,所以设置这个值改成默认1904年开始
useScientificFormatfalse数字转文本的时候在较大的数值的是否是否采用科学计数法
needHeadtrue是否需要写入头到excel
useDefaultStyletrue是否使用默认的样式
automaticMergeHeadtrue自动合并头,头中相同的字段上下左右都会去尝试匹配
excludeColumnIndexes需要排除对象中的index的数据
excludeColumnFieldNames需要排除对象中的字段的数据
includeColumnIndexes只要导出对象中的index的数据
includeColumnFieldNames只要导出对象中的字段的数据

WriteWorkbook

设置方法如下(找不到参数的看下通用参数里面是否存在):

EasyExcel.write(fileName, DemoData.class)
    // 在 write 方法之后, 在 sheet方法之前都是设置 WriteWorkbook 的参数
    .sheet("模板")
    .doWrite(() -> {
        // 分页查询数据
        return data();
    });

WriteWorkbook 方法:

public static ExcelWriterBuilder write()
public static ExcelWriterBuilder write(File file)
public static ExcelWriterBuilder write(String pathName)
public static ExcelWriterBuilder write(String pathName, Class head)
public static ExcelWriterBuilder write(OutputStream outputStream)
public static ExcelWriterBuilder write(OutputStream outputStream, Class head)
// 参数说明:
// 		file		创建的excel文件路径,路径需包含文件名
// 		pathName	创建的excel文件路径,路径需包含文件名
// 		outputStream	创建的excel文件的输出流,路径需包含文件名
// 		head		excel文件的行头实体类(标题栏)

WriteWorkbook 参数:

名称默认值描述
excelType当前excel的类型,支持XLS、XLSX、CSV
outputStreamfile二选一。写入文件的流
fileoutputStream二选一。写入的文件
templateInputStream模板的文件流
templateFile模板文件
charsetCharset#defaultCharset只有csv文件有用,写入文件的时候使用的编码
autoCloseStreamtrue自动关闭写入的流。
password读取文件的密码
inMemoryfalse是否在内存处理,默认会生成临时文件以节约内存。内存模式效率会更好,但是容易OOM
writeExcelOnExceptionfalse写入过程中抛出异常了,是否尝试把数据写入到excel

WriteSheet
  • 设置方法如下(找不到参数的看下通用参数里面是否存在)

    EasyExcel.write(fileName, DemoData.class)
        .sheet("模板")
        // 在 sheet 方法之后, 在 doWrite方法之前都是设置WriteSheet的参数
        .doWrite(() -> {
            // 分页查询数据
            return data();
        });
    

    参数说明:

    名称默认值描述
    sheetNo0需要写入的编码
    sheetName需要些的Sheet名称,默认同sheetNo

WriteTable
  • 设置方法如下(找不到参数的看下通用参数里面是否存在)

    EasyExcel.write(fileName, DemoData.class)
        .sheet("模板")
        .table()
        // 在 table 方法之后, 在 doWrite方法之前都是设置WriteTable的参数
        .doWrite(() -> {
            // 分页查询数据
            return data();
        });
    

    参数说明:

    名称默认值描述
    tableNo0需要写入的编码

代码示例

创建 Excel 文件

(1)配置实体类

在 EasyExcel 中,通过 java 代码生成或者解析 Excel 文件很简单。不需要创建繁琐的表头

  • 创建数据行的实体类 : ContractProductVo

  • 在实体类上通过注解配置 :标题,行高,列宽等数据

    • @ExcelProperty:在创建Excel的时候,自动的读取实体类中的注解配置,生成表头
  • 示例代码如下

    import com.alibaba.excel.annotation.ExcelProperty;
    import com.alibaba.excel.annotation.format.DateTimeFormat;
    import com.alibaba.excel.annotation.write.style.ColumnWidth;
    import com.alibaba.excel.annotation.write.style.ContentRowHeight;
    import com.alibaba.excel.annotation.write.style.HeadRowHeight;
    import lombok.Data;
    import java.util.Date;
    
    /**
     * Excel 实体类
     */
    @Data
    @ContentRowHeight(20)  //数据行高数
    @HeadRowHeight(20)     //表头高度
    @ColumnWidth(15)       //列宽
    public class UserExcelVo {
    
        @ExcelProperty("用户名")
        private String username;
    
        @ExcelProperty("密码")
        private String password;
    
        @ExcelProperty("电话")
        private String phone;
    
        @ExcelProperty("创建时间")
        @DateTimeFormat("yyyy-MM-dd")
        private Date createTime;
    }
    

(2)Excel 文件的生成和下载

  • 在 java 代码中通过 EasyExcel 工具类完成 excel 文件的生成和下载

  • 示例代码如下

    	/**
    	 * 使用 EasyExcel完成 excel 的生成和下载
    	 *  1、数据查询
    	 *  2、设置下载信息
    	 *  3、调用EasyExcel的工具类完成生成下载
    	 */
    	@RequestMapping("/printEasyExcel")
    	public void printEasyExcel(String inputDate) throws IOException {
    		//1、数据查询
    		List<ContractProductVo> list = contractService.findByShipTime(inputDate+"%");
    		//2、设置下载信息
    		response.setContentType("application/vnd.ms-excel"); //下载excel
    		response.setCharacterEncoding("utf-8");
    		String fileName = URLEncoder.encode("出货表", "UTF-8");
    		response.setHeader("Content-disposition", "attachment;filename=" + fileName + ".xlsx");
    		//3、调用EasyExcel的工具类完成生成下载
    		EasyExcel.write(response.getOutputStream())
    				.head(ContractProductVo.class) //设置表头
    				.sheet("test127") //指定页名称
    				.doWrite(list); //设置数据
    	}
    

模板打印

对于复杂样式的 Excel 文件创建,EasyExcel 支持模板打印。以一个 Excel 文件作为模板,自动的进行数据的拓展和填充。

  • 定义模板 ( 配置好了所有的样式 )
  • 配置实体类(表头)
  • java代码填充模板并下载 Excel

(1)定义模板

  • 在 easyExcel 中,支持两个模板语言
  • 语法:{map中的key}
    • 传入一个 map 集合:往往是用于处理非list集合的数据
  • 语法:{.对象中的属性名}
    • 传入一个 list 集合,会自动的循环 list 集合构造数据列表

在这里插入图片描述


(2)填充模板并下载 Excel 模板文件

	/**
	 * 模板打印
	 *
	 */
	@RequestMapping("/printTemplate")
	public void printTemplate(String inputDate) throws IOException {
		//1.准备数据
		List<ContractProductVo> list = contractService.findByShipTime(inputDate+"%");
		Map map = new HashMap<>();
		inputDate = inputDate.replaceAll("-0","-").replaceAll("-","年");
		map.put("time",inputDate);
		map.put("title1","客户名称");
		//2.设置下载信息
		response.setContentType("application/vnd.ms-excel"); //下载excel
		response.setCharacterEncoding("utf-8");
		String fileName = URLEncoder.encode("出货表", "UTF-8");
		response.setHeader("Content-disposition", "attachment;filename=" + fileName + ".xlsx");
		//3.加载excel模板
		String path = session.getServletContext().getRealPath("/")+"/make/tOUTPRODUCT.xlsx";
		//4.创建EasyExcel的excelWtire对象( 用于数据填充)
		ExcelWriter excelWriter = EasyExcel.write(response.getOutputStream())
				.head(ContractProductVo.class) //设置表头
				.withTemplate(path) //加载模板
				.build();
		//获取sheet对象
		WriteSheet sheet = EasyExcel.writerSheet().build();
		//5.调用方法完成填充map数据
		excelWriter.fill(map,sheet);
		//6.调用方法完成填充list数据
		excelWriter.fill(list,sheet);
		//7.属性资源,完成下载
		excelWriter.finish(); //下载excel文件,释放内存资源
	}


解析 Excel 文件

(1)配置实体类

创建实体类对象(用于解析封装对象),配置表头注解,EasyExcel 自动的根据注解配置获取文件中对应行的内容

  • @ExcelIgnoreUnannotated:忽略默认表头配置
  • @ExcelPropertym:配置表头
import cn.itcast.domain.BaseEntity;
import com.alibaba.excel.annotation.ExcelIgnoreUnannotated;
import com.alibaba.excel.annotation.ExcelProperty;

import java.io.Serializable;
import java.util.List;

/**
 * 合同下货物的实体类
 *  通过EasyExcel完成文件解析,在实体类上通过注解配置表头
 *  如果实体类中的属性,未配置@ExcelProperty
 *      默认按照属性的编写顺序,从excel中获取数据
 *      默认情况下,为配置注解的属性会影响数据封装
 *  在类上配置一个注解:忽略未配置注解的属性
 */
@ExcelIgnoreUnannotated
@Data
public class ContractProduct extends BaseEntity implements Serializable {

	private String id;

	@ExcelProperty("货号")
	private String productNo;		//货号
	private String productImage;	//图片路径
	@ExcelProperty("货物描述")
	private String productDesc;		//货描
	@ExcelProperty("装率")
	private String loadingRate;		//报运:装率    1/3
	@ExcelProperty("箱数")
	private Integer boxNum;			//报运:箱数    100
	@ExcelProperty("包装单位")
	private String packingUnit;		//包装单位:PCS/SETS   支/箱
	@ExcelProperty("数量")
	private Integer cnumber;		//数量                            300
	private Integer outNumber;		//报运:出货数量            200
	private Integer finished;		//报运:是否完成		no
	@ExcelProperty("要求")
	private String productRequest;	//要求
	@ExcelProperty("单价")
	private Double price;			//单价
	private Double amount;			//总金额,冗余
	private Integer orderNo;		//排序号
	private String contractId;      //合同号
	@ExcelProperty("生产厂家")
	private String factoryName;		//厂家名称,冗余字段
	private String factoryId;

	private List<ExtCproduct> extCproducts ;	//货物和附件,一对多
}


(2)解析 Excel 文件

修改 ContractProductController 对象,完成 EasyExcel 的文件批量上传

	/**
	 * 通过 EasyExcel 完成文件上传解析,批量货物保存
	 *  参数:购销合同id
	 *  参数:上传的excel文件对象
	 */
	@RequestMapping("/import")
	public String importExcel(String contractId, MultipartFile file) throws Exception {
		List<ContractProduct> list = EasyExcel.read(file.getInputStream())
				.head(ContractProduct.class) //设置表头,将数据转化为目标对象
				.sheet(0) //读取第一页数据
				.doReadSync(); //解析excel,获取所有的数据

		for (ContractProduct contractProduct : list) {
			System.out.println(contractProduct);
			contractProduct.setContractId(contractId);
			contractProduct.setCompanyId(getLoginCompanyId());
			contractProduct.setCompanyName(getLoginCompanyName());
		}
		contractProductService.saveAll(list);
		return  "redirect:/cargo/contractProduct/list.do?contractId="+contractId;
	}

拓展

csv 读写的支持

在3.0.0-beta1版本开始支持,读的时候会自动判断。写的时候指定 excelType 就行。

EasyExcel.write(fileName, DemoData.class)
   // 指定导出类型为csv
   .excelType(ExcelTypeEnum.CSV)
   .sheet("模板")
   .doWrite(() -> {
       // 分页查询数据
       return data();
   });

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2052720.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【Qt】常用控件QRadioButton

常用控件QRadioButton QRadioButton是单选按钮&#xff0c;可以在多个选项中选择一个。 作为QAbstractButton和QWidget的子类&#xff0c;其属性和用法&#xff0c;对于QRadioButton同样适用。 属性说明 checkable 是否能选中 checked 是否已经被选中. checkable 是 checked…

平安养老险宿州中支开展消防培训及逃生演练活动

近日&#xff0c;平安养老保险股份有限公司&#xff08;以下简称“平安养老险”&#xff09;宿州中心支公司邀请助安消防公司袁教官为公司员工开展消防安全知识培训。 培训教官结合近年来火灾典型案例&#xff0c;对火灾的危害、火灾的预防措施、灭火器材的使用进行深入浅出的…

centos 常用软件的安装和使用

redis 安装 0、下载、解压那些的我在这儿就不说了&#xff0c;有不懂的&#xff0c;看我之前写的博客。 1、在安装 redis 之前&#xff0c;我们得先安装它的依赖 yum install gcc-c 2、假设我下载放进的目录是 /usr/local/src/ &#xff0c; 并且 我的版本为 redis-6.0.8 …

数字化转型对金融服务业的影响

数字化转型正在塑造每个行业&#xff0c;从快速消费品到金融&#xff0c;每个行业都受到新兴技术的影响。 那么&#xff0c;数字化转型在金融服务中扮演什么角色&#xff1f;这对招聘前景有何影响&#xff1f; 我们探讨了数字化转型对该行业的影响、其对招聘策略的影响、数据…

适合大学生的耳机排行有哪些?四大TOP级性价比蓝牙耳机合集

大学生生活多彩多姿&#xff0c;耳机也成为了他们日常生活、学习和娱乐中不可或缺的配件。但面对市面上琳琅满目的耳机品牌和型号&#xff0c;选择性价比高且适合自身需求的耳机变得尤为重要&#xff0c;那么适合大学生的耳机排行有哪些&#xff1f;今天我就为大家精选了四大TO…

SpringBoot如何配置动态数据源?原理+实战

若没空探究原理可直接跳转到“实现方式&#xff1a;注解切面”目录 数据源切换方法 Spring对数据源的管理类似于策略模式&#xff0c;不懂策略模式也没关系&#xff0c;其实就是有一个全局的键值对&#xff0c;类型是Map<String, DataSource>。当JDBC操作数据库之时&am…

boot项目:程序包xxxx.xxxx.xxx不存在

你们好&#xff0c;我是金金金。 idea2021版本&#xff0c;在maven项目中运行程序的时候会出现java程序包不存在现象。(属于它的一个小bug) 场景 启动boot项目时报错如下 解决 清理缓存 重新加载所有maven项目 重新启动项目即可 测试 已测试&#xff0c;项目成功启动~ 编写有误…

代码规范 —— Redis 开发规范

优质博文&#xff1a;IT-BLOG-CN 一、开发规范 【1】弱依赖检查与线下确认&#xff1a;Redis必须是弱依赖&#xff0c;即Redis宕机不影响业务。包括超时检查。 【2】是否当存储使用检查&#xff1a;Redis不能作为存储设备来使用&#xff0c;只能作为缓存或状态等场景来使用。…

【Mudo库】实战项目之简要介绍

文章目录 前言一、效果演示二、模块1. 介绍2. 服务器模块3. 应用层模块 尾序 前言 各位C友们&#xff0c;好久不见&#xff0c;最近一个月在搞项目&#xff0c;算是半摆半学的状态吧&#xff0c;博客断更了一段时间&#xff0c;现在项目搞完了&#xff0c;博客之后也会慢慢更新…

机器学习周报(8.12-8.18)

文章目录 摘要Abstract1.Transformer的结构1.1 序列到序列&#xff08;Seq2seq&#xff09;的模型1.2 Transformer 结构1.2.1 Transformer 编码器&#xff08;Encoder&#xff09;1.2.2 Transformer解码器&#xff08;Decoder&#xff09;1.2.3 编码器-解码器注意力&#xff08…

Java语言程序设计——篇十四(2)

&#x1f33f;&#x1f33f;&#x1f33f;跟随博主脚步&#xff0c;从这里开始→博主主页&#x1f33f;&#x1f33f;&#x1f33f; 欢迎大家&#xff1a;这里是我的学习笔记、总结知识的地方&#xff0c;喜欢的话请三连&#xff0c;有问题可以私信&#x1f333;&#x1f333;&…

c++11(二)

一、右值引用 1、区分左值和右值 语法定义&#xff0c;左值可以取地址&#xff0c;右值无法取地址&#xff08;右值肯定有地址&#xff0c;但是为了和左值区分&#xff0c;语法上不让取地址&#xff09; 左值&#xff1a;一个表示数据的表达式&#xff08;变量名或解引用指针…

如何配置ESXI主机的IP地址管理

&#x1f3e1;作者主页&#xff1a;点击&#xff01; &#x1f427;Linux基础知识(初学)&#xff1a;点击&#xff01; &#x1f427;Linux高级管理防护和群集专栏&#xff1a;点击&#xff01; &#x1f510;Linux中firewalld防火墙&#xff1a;点击&#xff01; ⏰️创作…

通过python脚本查询自己阿里云账号里的某个域名的A记录解析情况,以及测拨,用于排查未使用的解析

安装sdk pip install aliyun-python-sdk-alidns代码全文 import json import requests from aliyunsdkcore.client import AcsClient from aliyunsdkalidns.request.v20150109 import DescribeDomainRecordsRequest# 替换为你的阿里云 AccessKey ID 和 AccessKey Secret acce…

栈与队列 - 逆波兰表达式求值

150. 逆波兰表达式求值 方法一&#xff1a;栈 /*** param {string[]} tokens* return {number}*/ var evalRPN function(tokens) {const stack [];for (const token of tokens) {if (isNaN(Number(token))) { // 非数字const n2 stack.pop(); // 出栈两个数字const n1 s…

假期作业--数据结构

1、顺序表实现学生管理系统&#xff08;参照顺序表技能&#xff09;写出菜单界面switch选择&#xff0c;功能1创建顺序表&#xff08;堆区&#xff09;&#xff0c;2录入学生信息&#xff0c;3插入一个学生信息&#xff0c;4删除一个学生信息&#xff0c;5按照位置修改一个学生…

javaEE中自定义注解以及注解的解析

注解&#xff1a; 就是java代码里的特殊标记&#xff0c;比如Override、Test,作用是&#xff1a;让其它程序根据注解信息来决定怎么执行程序。 自定义注解&#xff1a;自己定义注解 Public interface 注解名称{ Public 属性类型 属性名&#xff08;&#xff09; default 默认…

写字楼/办公室为什么要建设智慧公厕?有哪些价值?@卓振思众

智慧公厕是指利用先进技术和设备对公共厕所进行智能化管理的系统。这些技术包括物联网&#xff08;IoT&#xff09;、传感器技术、大数据分析和自动化系统等。【卓振思众】智慧公厕不仅提升了公厕的使用体验&#xff0c;还实现了更高效的管理和维护。 写字楼/办公室智慧公厕的定…

揭秘RAG与大模型对接:深入探讨9大隐藏挑战

前一段时间&#xff0c;各个大模型在争斗&#xff1a;谁能携带更长、更大的上下文 Prompt&#xff0c;比如 Kimi 说 200 万字&#xff0c;阿里通义千问又说自己能达 1000 万字&#xff1b;大家都知道 Prompt 很重要&#xff0c;但是 RAG 和 长的上下文文本携带 是两个不同的技术…

mac查看jdk安装目录

打开终端&#xff0c;直接输入命令&#xff1a; /usr/libexec/java_home终端即会输出jdk的安装目录&#xff1a;