锋盈数科-知识库 Logo
首页
软件开发
计算机基础
Hello Halo
新手必读
关于本知识库
登录 →
锋盈数科-知识库 Logo
首页 软件开发 计算机基础 Hello Halo 新手必读 关于本知识库
登录
  1. 首页
  2. 软件开发
  3. JAVA
  4. Spring Boot + EasyExcel实现Excel文件导入导出

Spring Boot + EasyExcel实现Excel文件导入导出

0
  • JAVA
  • 发布于 2024-08-16
  • 0 次阅读
黄健
黄健

本文由 简悦 SimpRead 转码, 原文地址 blog.csdn.net

Java 解析、生成 Excel 比较有名的框架有 Apache poi、jxl 等,使用者可自行斟酌。

一、 为什么使用 EasyExcel

1.1 内存控制

Apache poi、jxl 也能解析 Excel,但他们都存在一个问题就是非常的耗内存,poi 有一套 SAX 模式的 API 可以一定程度的解决一些内存溢出的问题,但 POI 还是有一些缺陷,比如 07 版 Excel 解压缩以及解压后存储都是在内存中完成的,内存消耗依然很大。

easyexcel 重写了 poi 对 07 版 Excel 的解析,一个 3M 的 excel 用 POI sax 解析依然需要 100M 左右内存,改用 easyexcel 可以降低到几 M,并且再大的 excel 也不会出现内存溢出;03 版依赖 POI 的 sax 模式,在上层做了模型转换的封装,让使用者更加简单方便。

1.2 使用简洁

EasyExcel 可以映射 excel 和实体类,让代码变的更加简洁,读写更方便。

二、springboot 整合 EasyExcel

2.1 添加 maven 依赖
<!-- spring-boot 使用自己项目的版本就好 -->
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-web</artifactId>
</dependency>
<!-- alibaba 相关依赖 -->
<dependency>
	<groupId>com.alibaba</groupId>
	<artifactId>easyexcel</artifactId>
	<version>3.3.3</version>
</dependency>
<dependency>
	<groupId>com.alibaba</groupId>
	<artifactId>fastjson</artifactId>
</dependency>
<!-- lombok -->
<dependency>
    <groupId>org.projectlombok</groupId>
    <artifactId>lombok</artifactId>
    <version>1.18.24</version>
</dependency>
2.2 读 Excel
2.2.1 准备数据源,Excel 表
姓名证件号证件类型
张三500000000000000000x户口簿
李四5111111111111111111身份证
王不二5222222222222222222身份证
赵五5252513xdg护照
2.2.2 创建与数据表对应的实体类
import com.alibaba.excel.annotation.ExcelProperty;
import com.alibaba.excel.annotation.format.DateTimeFormat;
import com.alibaba.excel.annotation.format.NumberFormat;
import lombok.Data;
import lombok.EqualsAndHashCode;

/**
 * @author lqf
 * @date 2024/1/11 11:00
 */
@Data
@EqualsAndHashCode
public class AccountCenterByExcelVO {
        /**
     * 姓名
     */
    // 可以使用指定 表格下标或者 表头名字 的方式来对应数据。视情况选择,优弊如下:
    // 使用表下标,会让制表变得不灵活,使用表名会造成 名字重复,只有一个字段读取到数据
    // @ExcelProperty(index = 1)
    @ExcelProperty("姓名")
    private String name;

    /**
     * 证件号
     */
    @ExcelProperty("证件号")
    private String IdNumber;

    /**
     * 证件类型
     */
    @ExcelProperty("证件类型")
    private String IdType;


    /**
     * 这里用string 去接日期才能格式化。我想接收年月日格式
     */
    @ExcelProperty("日期")
    @DateTimeFormat("yyyy年MM月dd日HH时mm分ss秒")
    private String date;

    /**
     * 想接收百分比的数字
     */
    @ExcelProperty("余额")
    @NumberFormat("#.##%")
    private String doubleData;
}
2.2.3 自定义转换器(可以不自定义,可使用 EasyExcel 自带的 PageReadListener)
import com.alibaba.excel.read.listener.ReadListener;
import com.alibaba.excel.context.AnalysisContext;
import com.alibaba.excel.util.ListUtils;
import com.alibaba.fastjson.JSON;
import com.insupro.flexibleServerJ.dto.resp.vo.AccountCenterByExcelVO;
import lombok.extern.slf4j.Slf4j;

import java.util.List;

/**
 * 有个很重要的点 DemoDataListener 不能被spring管理,要每次读取excel都要new,然后里面用到spring可以构造方法传进去
 * @author lqf
 */
@Slf4j
public class ACByExcelListener implements ReadListener<AccountCenterByExcelVO> {

    /**
     * 每隔 N 条 进行一次数据处理,根据实际使用情况处理,然后清理list ,方便内存回收
     */
    private static final int BATCH_COUNT = 100;
    /**
     * 缓存的数据
     */
    private List<AccountCenterByExcelVO> cachedDataList = ListUtils.newArrayListWithExpectedSize(BATCH_COUNT);

    /**
     * 假设这个是一个DAO,当然有业务逻辑这个也可以是一个service。
     * 用来存储或者处理数据
     */
    private AccountCenterByExcelVO excelVo;

    /**
     * 每次创建Listener的时候需要把spring管理的类传进来
     *
     * @param excelVo
     */
    public ACByExcelListener(AccountCenterByExcelVO excelVo) {
        this.excelVo = excelVo;
    }

    /**
     * 这个每一条数据解析都会来调用
     *
     * @param data
     * @param context
     */
    @Override
    public void invoke(AccountCenterByExcelVO data, AnalysisContext context) {
        cachedDataList.add(data);
        // 达到BATCH_COUNT了,需要去存储一次数据库,防止数据几万条数据在内存,容易OOM
        if (cachedDataList.size() >= BATCH_COUNT) {
            saveData();
            // 存储完成清理 list
            cachedDataList = ListUtils.newArrayListWithExpectedSize(BATCH_COUNT);
        }
    }

    /**
     * 所有数据解析完成了 都会来调用
     *
     * @param context
     */
    @Override
    public void doAfterAllAnalysed(AnalysisContext context) {
        // 这里也要保存数据,确保最后遗留的数据也存储到数据库
        saveData();
        log.info("所有数据解析完成!");
    }

    /**
     * 加上存储数据库
     */
    public void saveData() {
        log.info("{}条数据,开始处理数据!", cachedDataList.size());
        log.info("解析结果:"+JSON.toJSONString(cachedDataList));

        // 调用业务逻辑服务处理数据,实例是假装有业务逻辑
        log.info(excelVo.toString());

        log.info("数据处理成功!");
    }
}
2.2.4 创建一个文件处理 Controller

只是示例代码,业务逻辑没有写入 server 层,使用的时候自行将业务代码归类

import com.alibaba.excel.EasyExcel;
import com.alibaba.excel.read.listener.PageReadListener;
import com.alibaba.fastjson.JSON;
import com.insupro.flexibleServerJ.dto.resp.vo.AccountCenterByExcelVO;
import com.insupro.flexibleServerJ.utils.ACByExcelListener;
import lombok.extern.slf4j.Slf4j;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import org.springframework.web.multipart.MultipartFile;

import java.util.ArrayList;
import java.util.List;

/**
 * @author lqf
 * @date 2024/1/11 13:46
 */
@Slf4j
@RestController
@RequestMapping("api/easyExcel")
public class EasyExcelController {

    @RequestMapping("upExcel")
    public String upExcel(MultipartFile multipartFile){
        try {
            List<AccountCenterByExcelVO> acvoList = new ArrayList<>();
            // 第一种 使用 EasyExcel 自带的 PageReadListener,可以将数据读取后拿到读取的所有数据
            // 这里默认每次会读取100条数据 然后返回过来 直接调用使用数据就行
            // 具体需要返回多少行可以在`PageReadListener`的构造函数设置
            EasyExcel.read(multipartFile.getInputStream(), AccountCenterByExcelVO.class,
                    new PageReadListener<AccountCenterByExcelVO>(dateList ->{
                        // 这里默认每次会读取100条数据,然后进入下一次读取
                        log.info(JSON.toJSONString(dateList));
                        // 保存每一次的数据读取即可
                        acvoList.addAll(dateList);
                    }))
                    .sheet()
                    .doRead();
            System.out.println(acvoList);

            // 第二种,使用自定义监听器, 数据集处理需要再监听器中定义
            EasyExcel.read(multipartFile.getInputStream(), AccountCenterByExcelVO.class,
                            new ACByExcelListener(new AccountCenterByExcelVO()))
                    .sheet()
                    // 不添加属性,默认头部是1行
                    .headRowNumber(1)
                    .doRead();
        }catch (Exception e){
            e.printStackTrace();
        }
        return "完成";
    }
}
2.3 写出 excel
2.3.1 创建写出的 实体类
import com.alibaba.excel.annotation.ExcelProperty;
import com.alibaba.excel.annotation.format.DateTimeFormat;
import com.alibaba.excel.annotation.format.NumberFormat;
import lombok.Data;
import lombok.EqualsAndHashCode;

import java.util.Date;

/**
 * @author lqf
 * @date 2024/1/11 11:00
 */
@Data
@EqualsAndHashCode
public class AccountWriteForExcelVO {
    /**
     * 姓名
     * */
    // 可以使用指定 表格下标或者 表头名字 的方式来对应数据。视情况选择,优弊如下:
    // 使用表下标,会让制表变得不灵活,使用表名会造成 名字重复,只有一个字段读取到数据
    // @ExcelProperty(index = 1)
    @ExcelProperty("姓名")
    private String name;

    /**
     * 证件号
     */
    @ExcelProperty("证件号")
    private String IdNumber;

    /**
     * 证件类型
     */
    @ExcelProperty("证件类型")
    private String IdType;


    /**
     * 这里用string 去接日期才能格式化。我想接收年月日格式
     */
    @ExcelProperty("日期")
    @DateTimeFormat("yyyy年MM月dd日HH时mm分ss秒")
    private Date date;

    /**
     * 想接收百分比的数字
     */
    @ExcelProperty("余额")
    @NumberFormat("#.##%")
    private Double doubleData;
}
2.3.2 实现简单写出逻辑
import com.alibaba.excel.EasyExcel;
import com.alibaba.excel.ExcelWriter;
import com.alibaba.excel.util.ListUtils;
import com.alibaba.excel.write.metadata.WriteSheet;
import vo.AccountWriteForExcelVO;

import java.util.Date;
import java.util.List;

/**
 * @author lqf
 * @date 2024/1/11 11:00
 */
public class WriteExcleDemo {
    public static void main(String[] args) {
        // 使用自己指定输出的文件路径
        String fileName = "D:/workSpace/" + "simpleWrite" + System.currentTimeMillis() + ".xlsx";
        System.out.println(fileName);

        // 注意 数据量大参照 重复多次写入
        // 这里 需要指定写用哪个class去写,然后写到第一个sheet,名字为模板 然后文件流会自动关闭
        // 如果这里想使用03 则 传入excelType参数即可
        EasyExcel.write(fileName, AccountWriteForExcelVO.class)
                .sheet("模板")
                .doWrite(() -> {
                    // 分页查询数据
                    return data();
                });

        // 写法2
        // 这里 需要指定写用哪个class去写,然后写到第一个sheet,名字为模板 然后文件流会自动关闭
        // 如果这里想使用03 则 传入excelType参数即可
        EasyExcel.write(fileName, AccountWriteForExcelVO.class).sheet("模板").doWrite(data());

        // 写法3
        // 这里 需要指定写用哪个class去写
        try (ExcelWriter excelWriter = EasyExcel.write(fileName, AccountWriteForExcelVO.class).build()) {
            WriteSheet writeSheet = EasyExcel.writerSheet("模板").build();
            excelWriter.write(data(), writeSheet);
        }
    }

    private static List<AccountWriteForExcelVO> data() {
        List<AccountWriteForExcelVO> list = ListUtils.newArrayList();
        for (int i = 0; i < 10; i++) {
            AccountWriteForExcelVO data = new AccountWriteForExcelVO();
            data.setName("字符串" + i);
            data.setDate(new Date());
            data.setDoubleData(0.56);
            list.add(data);
        }
        return list;
    }
}

结语

更多使用方法,见官方文档
https://easyexcel.opensource.alibaba.com/docs/current/

标签: #软件开发 1171 #JAVA 991
相关文章

Spring 实现 3 种异步接口 2024-10-18 09:07

大家好,我是苏三~ 如何处理比较耗时的接口? 这题我熟,直接上异步接口,使用 Callable、WebAsyncTask 和 DeferredResult、CompletableFuture等均可实现。 但这些方法有局限性,处理结果仅返回单个值。在某些场景下,如果需要接口异步处理的同时,还持续不断地

重学SpringBoot3-集成Redis(五)之布隆过滤器 2024-10-08 11:24

更多SpringBoot3内容请关注我的专栏:《SpringBoot3》 期待您的点赞👍收藏⭐评论✍ 重学SpringBoot3-集成Redis(五)之布隆过滤器 1. 什么是布隆过滤器? * 基本概念 适用场景 2. 使用 Redis 实现布隆过滤器 * 项目依赖 Redis 配置

SpringBoot整合异步任务执行 2024-10-08 11:24

同步任务: 同步任务是在单线程中按顺序执行,每次只有一个任务在执行,不会引发线程安全和数据一致性等 并发问题 同步任务需要等待任务执行完成后才能执行下一个任务,无法同时处理多个任务,响应慢,影响用 户体验 异步任务: 异步任务是在多线程中同时执行,多个任务可以并发执行,同时处理多个请求,响应快,资源

springboot kafka多数据源,通过配置动态加载发送者和消费者 2024-10-08 11:24

前言 最近做项目,需要支持kafka多数据源,实际上我们也可以通过代码固定写死多套kafka集群逻辑,但是如果需要不修改代码扩展呢,因为kafka本身不处理额外逻辑,只是起到削峰,和数据的传递,那么就需要对架构做一定的设计了。 准备test kafka本身非常容易上手,如果我们需要单元测试,引入ja

SpringBoot 集成 Redis 2024-10-08 11:24

一:SpringBoot 集成 Redis ①Redis是一个 NoSQL(not only)数据库, 常作用缓存 Cache 使用。 ②Redis是一个中间件、是一个独立的服务器;常用的数据类型: string , hash ,set ,zset , list ③通过Redis客户端可以使用多种语

SpringBoot整合QQ邮箱 2024-10-08 11:24

SpringBoot可以通过导入依赖的方式集成多种技术,这当然少不了我们常用的邮箱,现在本章演示SpringBoot整合QQ邮箱发送邮件…. 下面按步骤进行: 1.获取QQ邮箱授权码 1.1 登录QQ邮箱 1.2 开启SMTP服务 找到下图中的SMTP服务区域,如果当前账号未开启的话自己手动开启。

目录

IT 外包服务商

  • 意见投递
  • zyf6619

软件开发应用

主菜单

  • 首页
  • 软件开发
  • 计算机基础
  • Hello Halo
  • 新手必读
  • 关于本知识库
Copyright © 2024 your company All Rights Reserved. Powered by Halo.