Springboot集成Hutool导出CSV

1. 引入依赖

xml 复制代码
 <dependency>
     <groupId>cn.hutool</groupId>
     <artifactId>hutool-all</artifactId>
     <version>5.8.11</version>
 </dependency>

2. CSV工具类

java 复制代码
import cn.hutool.core.text.csv.CsvUtil;
import cn.hutool.core.text.csv.CsvWriter;
import lombok.extern.slf4j.Slf4j;

import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.OutputStream;
import java.io.OutputStreamWriter;
import java.lang.reflect.Field;
import java.net.URLEncoder;
import java.nio.charset.StandardCharsets;
import java.util.List;
import java.util.Map;
import java.util.zip.ZipEntry;
import java.util.zip.ZipOutputStream;

@Slf4j
public class CsvUtils {

    public static <T> void export(HttpServletResponse response, String fileName, List<T> dataList) throws IOException {
        // 设置响应内容类型
        response.setContentType("text/csv");
        response.setCharacterEncoding(StandardCharsets.UTF_8.name());
        fileName = URLEncoder.encode(fileName, StandardCharsets.UTF_8);
        response.setHeader("Content-Disposition", "attachment;filename=\"" + fileName + ".csv\"");
        try (OutputStream out = response.getOutputStream();
             OutputStreamWriter writer = new OutputStreamWriter(out, StandardCharsets.UTF_8)) {
            // 写入UTF-8 BOM,帮助Notepad识别
            writer.write("\uFEFF");
            // 使用HuTool的CsvWriter
            CsvWriter csvWriter = CsvUtil.getWriter(writer);
            for (T data : dataList) {
                String[] row = convertToRow(data);
                csvWriter.write(row);
            }
            // 确保所有数据写入输出流
            csvWriter.flush();
        }
    }

    public static <T> void exportZip(HttpServletResponse response, Map<Integer, List<T>> dataLists) throws IOException {
        // 设置响应内容类型
        response.setContentType("application/zip");
        response.setCharacterEncoding(StandardCharsets.UTF_8.name());
        String zipFileName = URLEncoder.encode("地区结果.zip", StandardCharsets.UTF_8);
        response.setHeader("Content-Disposition", "attachment;filename=\"" + zipFileName + "\"");

        // 创建ZIP输出流
        try (OutputStream out = response.getOutputStream();
             ZipOutputStream zipOut = new ZipOutputStream(out, StandardCharsets.UTF_8)) {

            for (Map.Entry<Integer, List<T>> listEntry : dataLists.entrySet()) {
                Integer fileName = listEntry.getKey();
                List<T> dataList = listEntry.getValue();
                // 创建ZIP条目
                ZipEntry zipEntry = new ZipEntry(fileName + ".csv");
                zipOut.putNextEntry(zipEntry);

                // 创建CSV写入器
                OutputStreamWriter writer = new OutputStreamWriter(zipOut, StandardCharsets.UTF_8);
                writer.write("\uFEFF");

                CsvWriter csvWriter = CsvUtil.getWriter(writer);

                // 写入数据
                for (T data : dataList) {
                    String[] row = convertToRow(data);
                    csvWriter.write(row);
                }
                csvWriter.flush(); // 确保所有数据写入输出流
                zipOut.closeEntry(); // 关闭当前条目
            }

        }
    }

    private static <T> String[] convertToRow(T data) {
        Field[] fields = data.getClass().getDeclaredFields();
        String[] row = new String[fields.length];

        try {
            for (int i = 0; i < fields.length; i++) {
                // 设置字段可访问
                fields[i].setAccessible(true);
                // 获取字段值
                Object value = fields[i].get(data);
                // 转换为字符串,处理null
                row[i] = value != null ? value.toString() : "";
            }
        } catch (IllegalAccessException e) {
            log.error("数据转换异常:{}",e.getMessage());
        }
        return row;
    }
}

3. 接口

java 复制代码
@Slf4j
@RestController
public class RegionExportController {

    @SneakyThrows
    @PostMapping("/api/v1/regionExport")
    public void exportVehicleResult(HttpServletResponse response) {
        // 模拟数据
        List<RegionImportDTO> list = RegionImportDTO.exportData();
        CsvUtils.export(response, "区域结果表", list);
    }


    @SneakyThrows
    @PostMapping("/api/v1/regionExportZip")
    public void exportZipVehicleResult(HttpServletResponse response) {
        // 模拟数据
        List<RegionImportDTO> list = RegionImportDTO.exportData();
        Map<Integer, List<RegionImportDTO>> listMap = list.stream().collect(Collectors.groupingBy(RegionImportDTO::getType));
        CsvUtils.exportZip(response, listMap);
    }
}
相关推荐
IT 行者15 分钟前
GitHub Spec Kit 实战(五):/speckit.tasks 怎么拆——Spec Kit 五部曲收官
java·ai编程·claude
提笔了无痕24 分钟前
如何用Go实现整套RAG流程
开发语言·后端·golang
(Charon)26 分钟前
【C++ 面试高频基础:指针、引用、const、static、new/delete 总结】
java·开发语言
成都第一深情IZZO40 分钟前
事务未提交就发送 MQ,导致消费者读不到订单数据的问题
后端
Yeats_Liao44 分钟前
Feed流系统设计(三):数据模型与存储设计,从表结构到Redis收件箱
java·javascript·redis
大橙子打游戏1 小时前
Fable5不能用了,但是依然能让 AI 纯靠截图玩通宝可梦
后端
JiaHao汤1 小时前
分布式事务方案全景:从理论到 Seata 落地
java·分布式·spring·spring cloud
Jason_chen1 小时前
Linux 3.0 总线机制与故障排查详解
后端
成都第一深情IZZO1 小时前
Spring Boot 动态数据源在事务中切库失效问题排查
后端
_遥远的救世主_1 小时前
稳定性工程:SLO 量化、降级收敛与故障兜底体系
后端