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

动态表头导出EasyExcel

在 Spring Boot 中结合 EasyExcel 实现动态表头导出(无实体类,表头和字段(前端传表名,字段值动态查询,返回List<Map<String,Object>>)由前端传递)可以通过以下步骤实现。以下是完整示例:


1. 前端请求数据结构

假设前端传递的 JSON 格式如下:

{
  "headers": [
    {"title": "姓名", "field": "name"},
    {"title": "年龄", "field": "age"},
    {"title": "城市", "field": "city"}
  ],
  "data": [
    {"name": "张三", "age": 25, "city": "北京"},
    {"name": "李四", "age": 30, "city": "上海"}
  ]
}

2. 后端 DTO 定义

定义接收参数的 DTO 类:

@Data
public class ExportRequest {
    private List<Header> headers;
    private List<Map<String, Object>> data;

    @Data
    public static class Header {
        private String title;  // 表头名称
        private String field;  // 数据字段名
    }
}

3. Controller 层接口

处理导出请求:

@RestController
public class ExportController {

    @Autowired
    private ExportService exportService;

    @PostMapping("/export")
    public void exportExcel(@RequestBody ExportRequest request, HttpServletResponse response) {
        exportService.export(request, response);
    }
}

4. Service 层实现

核心导出逻辑:

@Service
public class ExportService {

    public void export(ExportRequest request, HttpServletResponse response) {
        try {
            // 设置响应头
            response.setContentType("application/vnd.openxmlformats-officedocument.spreadsheetml.sheet");
            response.setCharacterEncoding("UTF-8");
            String fileName = URLEncoder.encode("动态导出.xlsx", "UTF-8");
            response.setHeader("Content-Disposition", "attachment; filename=" + fileName);

            // 获取输出流
            OutputStream outputStream = response.getOutputStream();

            // 动态构建表头和数据
            WriteSheet writeSheet = EasyExcel.writerSheet("Sheet1").build();
            ExcelWriter excelWriter = EasyExcel.write(outputStream).build();

            // 动态添加表头
            WriteTable writeTable = new WriteTable();
            List<List<String>> head = buildHead(request.getHeaders());
            writeTable.setHead(head);

            // 动态填充数据
            List<List<Object>> data = buildData(request.getHeaders(), request.getData());
            excelWriter.write(data, writeSheet, writeTable);

            // 关闭流
            excelWriter.finish();
            outputStream.flush();
        } catch (IOException e) {
            throw new RuntimeException("导出失败", e);
        }
    }

    // 构建表头
    private List<List<String>> buildHead(List<ExportRequest.Header> headers) {
        List<List<String>> head = new ArrayList<>();
        for (ExportRequest.Header header : headers) {
            List<String> columnHead = Collections.singletonList(header.getTitle());
            head.add(columnHead);
        }
        return head;
    }

    // 构建数据行
    private List<List<Object>> buildData(List<ExportRequest.Header> headers, List<Map<String, Object>> dataList) {
        List<List<Object>> data = new ArrayList<>();
        for (Map<String, Object> rowData : dataList) {
            List<Object> row = new ArrayList<>();
            for (ExportRequest.Header header : headers) {
                row.add(rowData.get(header.getField()));
            }
            data.add(row);
        }
        return data;
    }
}

5. 关键点说明

  1. 动态表头

    • 通过 buildHead() 方法将前端传递的 headers 转换为 EasyExcel 需要的 List<List<String>> 格式。
  2. 动态数据

    • 通过 buildData() 方法,根据 headers 中定义的 field 字段顺序,从 data 中提取对应值,构建数据行。
  3. 流式导出

    • 使用 ExcelWriter 直接操作输出流,避免内存溢出(适合大数据量)。

6. 测试与验证

使用 Postman 发送请求:

  • URL: POST http://localhost:8080/export

  • Body(JSON):

    {
      "headers": [
        {"title": "姓名", "field": "name"},
        {"title": "年龄", "field": "age"},
        {"title": "城市", "field": "city"}
      ],
      "data": [
        {"name": "张三", "age": 25, "city": "北京"},
        {"name": "李四", "age": 30, "city": "上海"}
      ]
    }
    
  • 响应:浏览器自动下载 动态导出.xlsx,内容如下:

    姓名年龄城市
    张三25北京
    李四30上海

7. 扩展优化

  • 字段校验:确保前端传递的 fielddata 中存在对应值。
  • 大数据量分页:如果数据量过大,可分页查询后分批写入。
  • 自定义样式:通过 WriteHandler 动态设置单元格样式(如字体、颜色)。

通过这种方式,无需定义实体类即可实现完全动态的 Excel 导出功能,表头和字段完全由前端控制。

相关文章:

  • 上海最新资讯重庆seo什么意思
  • 建网站开发国外客户企业建站都有什么网站
  • 做兼职的网站都有哪些工作内容视频网站建设
  • 湖南做网站 真好磐石网络如何建立公司网站网页
  • 网站做国际化小程序开发系统
  • 专业团队图片高清北京seo软件
  • 基于C语言对CAPL语法基础的理解
  • 天梯赛:L2-001 紧急救援
  • 6.6.5 SQL访问控制
  • 第16届蓝桥杯模拟赛题解 第三场 Java
  • stm32使用(无线串口)实现收发、判断数据+DMA(HAL库)
  • 2022年全国职业院校技能大赛网络系统管理赛项模块A:网络构建(样题10)-网络部分解析-附详细代码
  • 跨端方案选型:对比Uni-app与Taro在复杂电商项目中的技术选型依据参考
  • 大白话解释负载均衡Nginx是什么 有什么用 怎么用#
  • ClkLog里程碑:荣获2024上海开源技术应用创新竞赛三等奖
  • 记录深度学习中有用的终端命令
  • 第三章 组件(11)- 动态组件与表格组件
  • 【Qt】MVC设计模式
  • Java线程池
  • LLVM - 编译器前端 - 将源文件转换为抽象语法树
  • 在docker容器中运行Ollama部署deepseek-r1大模型
  • C# String 常用操作方法详解
  • 检查SSH安全配置-sshd服务端未认证连接最大并发量配置
  • React Native 核心技术知识点快速入门
  • 用大白话解释日志处理Log4j 是什么 有什么用 怎么用
  • 45.matlab产生正弦叠加信号并保存为txt文本