使用EasyExcel进行简单的导入、导出

news2025/3/18 13:45:59

准备

pom.xml添加依赖

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

导入

controller

// 用户导入
 @Operation(summary = "用户导入")
 @PostMapping("/importUserExcel")
 @ResponseBody
 public Result importUserExcel(@RequestParam("file")MultipartFile file){
     return userService.importUserExcel(file);
 }

导入实体类

@Data
public class UserImportParam {

    @ExcelProperty(value = "* 所属组织code", index = 0)
    private String orgCode;

    @ExcelProperty(value = "* 用户编号", index = 1)
    private String code;

    @ExcelProperty(value = "* 用户名", index = 2)
    private String name;

    @ExcelProperty(value = "* 邮箱", index = 3)
    private String email;

    @ExcelProperty(value = "* 手机号", index = 4)
    private String phoneNumber;

    @ExcelProperty(value = "* 性别", index = 5)
    private String gender;

    @ExcelProperty(value = "* 角色", index = 6)
    private String isStaff;

    @ExcelProperty(value = "简介", index = 7)
    private String userInfo;
}

通过@ExcelProperty进行实体类与Excel文件里标题进行映射,如果只存在value,必须与Excel标题一一对应。如果两者都存在index会覆盖value的匹配逻辑时

数据解析监听

public class UserImportListener extends AnalysisEventListener<UserImportParam> {

    private final UserServiceImpl userService;
    // 构造器注入
    public UserImportListener(UserServiceImpl userService) {
        this.userService = userService;
    }

    private static final int BATCH_SIZE = 100;
    private List<SysUser> cachedList = new ArrayList<>(BATCH_SIZE);

    // 用于校验
    @Override
    public void invoke(UserImportParam data, AnalysisContext context) {
        // 校验用户所属组织是否存在
        if (StringUtils.isEmpty(data.getOrgCode())) {
            throw new RuntimeException("所属组织不能为空");
        }
        // 生成实体类
        SysUser user = new SysUser();
        cachedList.add(user);
    }

    /**
     * 解析后的操作
     */
    @Override
    public void doAfterAllAnalysed(AnalysisContext context) {
        if (!cachedList.isEmpty()) {
            userService.saveBatch(cachedList);
        }
    }
}

service

 @Transactional
 public Result importUserExcel(MultipartFile file) {
     try {
         InputStream inputStream = file.getInputStream();
         EasyExcel.read(inputStream, UserImportParam.class, new UserImportListener(this))
                 .sheet()
                 .doRead();
         return Result.success("导入成功");
     } catch (Exception e) {
         TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
         return Result.error("导入失败");
     }
 }

导出

基础导出

import com.alibaba.excel.EasyExcel;
import jakarta.servlet.http.HttpServletResponse;

import java.io.IOException;
import java.net.URLEncoder;
import java.util.List;

public class EasyExcelUtil {
    /**
     * 导出 Excel 文件
     *
     * @param response HttpServletResponse 对象
     * @param fileName 导出的文件名
     * @param dataList 要导出的数据列表
     * @param clazz    数据列表元素的类型
     * @throws IOException 当写入输出流发生错误时抛出此异常
     */
    public static <T> void exportExcel(HttpServletResponse response, String fileName, List<T> dataList, Class<T> clazz) throws IOException, IOException {
        // 设置响应头
        response.setContentType("application/vnd.openxmlformats-officedocument.spreadsheetml.sheet");
        response.setCharacterEncoding("utf-8");
        fileName = URLEncoder.encode(fileName, "UTF-8").replaceAll("\\+", "%20");
        response.setHeader("Content-disposition", "attachment;filename*=utf-8''" + fileName);

        // 导出数据到 Excel
        EasyExcel.write(response.getOutputStream(), clazz).sheet("数据信息").doWrite(dataList);
    }
}
@Service
public class UserService {
    @Autowired
    private UserRepository userRepository;

    public void exportUserExcel(HttpServletResponse response) throws IOException {
        // 从数据库查询所有用户信息
        List<User> userList = userRepository.findAll();
        // 调用工具类导出数据
        EasyExcelUtil .exportExcel(response, "用户信息.xlsx", userList, User.class);
    }
} 
 // 用户信息导出
 @Operation(summary = "用户信息导出")
 @PostMapping("/exportUserExcel")
 public void exportUserExcel( HttpServletResponse response) throws IOException {
    userService.exportUserExcel(response);
 }
@Data
public class User{
    @ExcelProperty("用户名")
    private String name; // 对应数据库字段 name

    @ExcelProperty("年龄")
    private Integer age; // 对应数据库字段 age

    @ExcelProperty("注册时间")
    private LocalDateTime createTime; // 对应数据库字段 create_time
}

@ExcelProperty 的 value 可以设置导出的列头

动态列导出

/**
 * 动态导出 Excel 文件
 *
 * @param response   HttpServletResponse 对象
 * @param fileName   导出的文件名
 * @param dataList   要导出的数据列表
 * @param fieldNames 要导出的字段名
 * @param clazz      数据列表元素的类型
 * @throws IOException 当写入输出流发生错误时抛出此异常
 */
public static <T> void exportExcel(HttpServletResponse response, String fileName, List<T> dataList, List<String> fieldNames, Class<T> clazz) throws Exception {
    // 设置响应头
    response.setContentType("application/vnd.openxmlformats-officedocument.spreadsheetml.sheet");
    response.setCharacterEncoding("utf-8");
    fileName = URLEncoder.encode(fileName, "UTF-8").replaceAll("\\+", "%20");
    response.setHeader("Content-disposition", "attachment;filename*=utf-8''" + fileName);
    // 生成表头
    List<List<String>> head = generateHead(fieldNames, clazz);
    // 生成数据
    List<List<Object>> data = generateData(dataList, fieldNames, clazz);
    // 导出数据到 Excel
    EasyExcel.write(response.getOutputStream()).head(head).sheet("数据信息").doWrite(data);
}

// 生成表头
private static List<List<String>> generateHead(List<String> fieldNames, Class<?> clazz) throws Exception {
    List<List<String>> head = new ArrayList<>();
    for (String fieldName : fieldNames) {
        // 获取字段的注解
        Field field = clazz.getDeclaredField(fieldName);
        // 获取字段的注解
        ExcelProperty annotation = field.getAnnotation(ExcelProperty.class);
        // 如果注解为空,则使用字段名作为列名
        String columnName = annotation != null ? annotation.value()[0] : fieldName;
        // 将列名添加到表头中
        head.add(Collections.singletonList(columnName));
    }
    return head;
}

// 生产数据
private static List<List<Object>> generateData(List<?> dataList, List<String> fieldNames, Class<?> clazz) throws Exception {
    List<List<Object>> data = new ArrayList<>();
    for (Object item : dataList) {
        List<Object> row = new ArrayList<>();
        for (String fieldName : fieldNames) {
            // 获取字段的值
            Field field = clazz.getDeclaredField(fieldName);
            // 设置字段可访问
            field.setAccessible(true);
            // 设置字段的值
            row.add(field.get(item));
        }
        data.add(row);
    }
    return data;
}
// 动态导出
List<String> fieldsToExport = Arrays.asList("code", "name", "gender", "isStaff");
EasyExcelUtil.exportExcel(response, "用户信息.xlsx", userList, fieldsToExport, UserExportResult.class);

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

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

相关文章

Excel(函数篇):COUNTIF与CONUTIFS函数、SUMIF与SUMIFS函数、ROUND函数、MATCH与INDEX函数、混合引用与条件格式

目录 COUNTIF和COUNTIFS函数COUNTIF函数COUNTIFS函数SUMIF和SUMIFS函数SUMIF函数SUMIFS函数SUMIFS函数与控件实现动态年月汇总ROUND、ROUNDUP、ROUNDDOWN函数单元格混合引用条件格式与公式,标记整行数据MATCH和INDEX函数COUNTIF和COUNTIFS函数 COUNTIF函数 统计下“苏州”出现…

虚拟定位 1.2.0.2 | 虚拟定位,上班打卡,校园跑步模拟

Fake Location是一款运行于安卓平台上的功能强大、简单实用的虚拟定位软件。它能够帮助用户自定义位置到地图上的任意地方&#xff0c;以ROOT环境运行不易被检测&#xff0c;同时也支持免ROOT运行。提供路线模拟、步频模拟、WIFI模拟等方式&#xff0c;支持反检测。 大小&…

【最大异或和——可持久化Trie】

题目 代码 #include <bits/stdc.h> using namespace std;const int N 6e510; //注意这里起始有3e5&#xff0c;又可能插入3e5 const int M N * 25;int rt[N], tr[M][2]; //根&#xff0c;trie int idx, cnt, br[M]; //根分配器&#xff0c;点分配器&#xff0c;点的相…

C# WPF编程-启动新窗口

C# WPF编程-启动新窗口 新建窗口&#xff1a; 工程》添加》窗口 命名并添加新的窗口 这里窗口名称为Window1.xaml 启动新窗口 Window1 win1 new Window1(); win1.Show(); // 非模态启动窗口win1.ShowDialog(); // 模态启动窗口 模态窗口&#xff1a;当一个模态窗口被打开时&a…

Python 实现大文件的高并发下载

项目背景 基于一个 scrapy-redis 搭建的分布式系统&#xff0c;所有item都通过重写 pipeline 存储到 redis 的 list 中。这里我通过代码演示如何基于线程池 协程实现对 item 的中文件下载。 Item 结构 目的是为了下载 item 中 attachments 保存的附件内容。 {"crawl_tim…

【最新】 ubuntu24安装 1panel 保姆级教程

系统&#xff1a;ubuntu24.04.1 安装软件 &#xff1a;1panel 第一步&#xff1a;更新系统 sudo apt update sudo apt upgrade 如下图 第二步&#xff1a;安装1panel&#xff0c;运行如下命令 curl -sSL https://resource.fit2cloud.com/1panel/package/quick_start.sh -o …

c++图论(二)之图的存储图解

在 C 中实现图的存储时&#xff0c;常用的方法包括 邻接矩阵&#xff08;Adjacency Matrix&#xff09;、邻接表&#xff08;Adjacency List&#xff09; 和 边列表&#xff08;Edge List&#xff09;。以下是具体实现方法、优缺点分析及代码示例&#xff1a; 1. 邻接矩阵&…

c++图论(一)之图论的起源和图的概念

C 图论之图论的起源和图的概念 图论&#xff08;Graph Theory&#xff09;是数学和计算机科学中的一个重要分支&#xff0c;其起源可以追溯到 18 世纪 的经典问题。以下是图论的历史背景、核心起源问题及其与基本概念和用途&#xff1a; 借用一下CSDN的图片哈 一、图论的起源&…

ChatGPT and Claude国内使用站点

RawChat kelaode chatgptplus chatopens&#xff08;4.o mini免费&#xff0c;plus收费&#xff09; 网页&#xff1a; 定价&#xff1a; wildcard 网页&#xff1a; 虚拟卡定价&#xff1a; 2233.ai 网页&#xff1a; 定价&#xff1a; MaynorAPI chatgpt cla…

进行性核上性麻痹:精心护理,点亮希望之光

进行性核上性麻痹是一种罕见的神经退行性疾病&#xff0c;严重影响患者的生活质量。有效的健康护理能够在一定程度上缓解症状、延缓病情发展&#xff0c;给患者带来更好的生活体验。 在日常生活护理方面&#xff0c;由于患者平衡能力逐渐下降&#xff0c;行动不便&#xff0c;居…

ZED X系列双目3D相机的耐用性与创新设计解析

在工业自动化和学术研究领域&#xff0c;高精度的视觉设备正成为提升效率和质量的关键。ZED X系列AI立体相机&#xff0c;凭借其先进的技术和耐用的设计&#xff0c;为这一领域带来了新的可能。 核心技术&#xff1a;深度感知与精准追踪 ZED X系列的核心技术之一是Neural Dept…

HarmonyOS三层架构实战

目录&#xff1a; 1、三层架构项目结构1.0、三层架构简介1.1、 common层&#xff08;主要放一些公共的资源等&#xff09;1.2、 features层&#xff08;主要模块定义的组件以及图片等静态资源&#xff09;1.3、 products层&#xff08;主要放主页面层和一些主要的资源&#xff…

计算机四级 - 数据库原理 - 第4章 「关系数据库标准语言SQL」

4.1 SQL概述 4.1.1 结构化查询语言SQL SQL(Structured Query Language)称为结构化查询语言&#xff0c;它是由1974年由Boyce和Chamberi提出的&#xff0c;1975年至1979年IBM公司的San Jose Research Laboratory研制了关系数据库管理系统的原型系统System R,并实现了这种语198…

基于PMU的14节点、30节点电力系统状态估计MATLAB程序

“电气仔推送”获得资料&#xff08;专享优惠&#xff09; 程序简介&#xff1a; 程序采用三种方法对14节点和30节点电力系统状态进行评估&#xff1a; ①PMU同步向量测量单元结合加权最小二乘法&#xff08;WLS&#xff09;分析电力系统的电压幅值和相角状态&#xff1b; …

Deepseek API+Python测试用例一键生成与导出-V1.0.2【实现需求文档图片识别与用例生成自动化】

在测试工作中&#xff0c;需求文档中的图片&#xff08;如界面设计图、流程图&#xff09;往往是测试用例生成的重要参考。然而&#xff0c;手动提取图片并识别内容不仅耗时&#xff0c;还容易出错。本文将通过一个自研小工具&#xff0c;结合 PaddleOCR 和大模型&#xff0c;自…

整形在内存中的存储(例题逐个解析)

目录 一.相关知识点 1.截断&#xff1a; 2.整形提升&#xff1a; 3.如何 截断&#xff0c;整型提升&#xff1f; &#xff08;1&#xff09;负数 &#xff08;2&#xff09;正数 &#xff08;3&#xff09;无符号整型&#xff0c;高位补0 注意&#xff1a;提升后得到的…

蓝牙系统的核心组成解析

一、硬件层&#xff1a;看得见的物理载体 1. 射频模块&#xff08;Radio Frequency Module&#xff09; 专业描述&#xff1a;工作在2.4GHz ISM频段&#xff0c;支持GFSK/π/4 DQPSK/8DPSK调制方式 功能类比&#xff1a;相当于人的"嘴巴"和"耳朵" 发射端…

uniapp笔记-底部和首部标签页菜单生成

逻辑 这些都是需要配置pages.json文件。 其中底部需要手动配置tarBar&#xff0c;如&#xff1a; "tabBar": {"list":[{"pagePath": "pages/index/index","text": "首页"},{"pagePath": "pages/…

SpringBoot 和vue前后端配合开发网页拼图10关游戏源码技术分享

今天分享一个 前后端结合 的网页游戏 开发项目源码技术。 这也是我第一次写游戏类的程序&#xff0c;虽然不是特别复杂的游戏&#xff0c;但是是第一次写&#xff0c;肯定要记录一下了&#xff0c;哈哈。 游戏的内容 就是 我们显示中玩的那个 拼图碎片的 游戏&#xff0c;类似下…

OpenCV计算摄影学(21)非真实感渲染之边缘保留滤波器edgePreservingFilter()

操作系统&#xff1a;ubuntu22.04 OpenCV版本&#xff1a;OpenCV4.9 IDE:Visual Studio Code 编程语言&#xff1a;C11 算法描述 滤波是图像和视频处理中的基础操作。边缘保留平滑滤波器被广泛应用于多种不同场景[98]。 cv::edgePreservingFilter 是一种边缘保留滤波器&#…