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);
    }
}
相关推荐
专注API从业者1 小时前
Open Claw 京东商品监控选品实战:一键抓取、实时监控、高效选品
java·服务器·数据库
摇滚侠1 小时前
DBeaver 导入数据库 导入 SQL 文件 MySQL 备份恢复
java·数据库·mysql
古城小栈1 小时前
从 cargo-whero 库中,找到提升 rust 的契机
开发语言·后端·rust
keep one's resolveY1 小时前
SpringBoot实现重试机制的四种方案
java·spring boot·后端
天空属于哈夫克32 小时前
企业微信API常见的错误和解决方案
java·数据库·企业微信
摇滚侠3 小时前
VMvare 虚拟机 Oracle19c 安装步骤,远程连接 Oracle19c,百度网盘安装包
java·oracle
梁萌3 小时前
idea报错找不到XX包的解决方法
java·intellij-idea·启动报错·缺少包
Agent产品评测局3 小时前
生产排期与MES/ERP系统打通,实操方法详解 —— 2026企业级智能体自动化选型与实战指南
java·运维·人工智能·ai·chatgpt·自动化
阿丰资源3 小时前
基于Spring Boot的电影城管理系统(直接运行)
java·spring boot·后端
IT_陈寒3 小时前
SpringBoot自动配置的坑差点让我加班到天亮
前端·人工智能·后端