Spring Boot中实现多文件打包下载

在Spring Boot中实现多文件打包下载,主要通过将多个文件压缩为ZIP格式并返回给客户端。以下是两种典型实现方案:


​一、本地文件打包下载​

​核心步骤​

  1. ​接收文件列表参数​
    通过@RequestParam List<String> filenames接收前端传递的文件名列表。
  2. ​动态生成ZIP文件​
    使用ZipOutputStream将文件逐个写入内存流,避免生成物理文件。
  3. ​设置响应头​
    通过Content-Disposition指定文件名,Content-Type设置为application/zip

​代码示例​

scss 复制代码
@GetMapping("/download-multiple")
public ResponseEntity<byte[]> downloadMultipleFiles(@RequestParam List<String> filenames) {
    try (ByteArrayOutputStream baos = new ByteArrayOutputStream();
         ZipOutputStream zos = new ZipOutputStream(baos)) {
         
        for (String filename : filenames) {
            Path filePath = Paths.get("/path/to/files").resolve(filename).normalize();
            if (Files.exists(filePath)) {
                ZipEntry zipEntry = new ZipEntry(filename);
                zos.putNextEntry(zipEntry);
                Files.copy(filePath, zos);
                zos.closeEntry();
            }
        }
        zos.finish();

        HttpHeaders headers = new HttpHeaders();
        headers.add(HttpHeaders.CONTENT_DISPOSITION, "attachment; filename="files.zip"");
        return ResponseEntity.ok().headers(headers).body(baos.toByteArray());
    } catch (IOException e) {
        return ResponseEntity.internalServerError().build();
    }
}

​关键点​​:

  • 使用try-with-resources自动关闭流,避免资源泄漏。
  • normalize()方法防止路径遍历攻击(如../../)。
  • 内存中生成ZIP,适合小文件场景。

​二、云存储(MinIO)集成方案​

若文件存储在云存储(如MinIO),需先获取文件流再压缩:

​实现步骤​

  1. ​配置MinIO客户端​

    添加依赖并配置MinIO连接信息:

    yaml 复制代码
    minio:
      url: http://localhost:9000
      accessKey: your-key
      secretKey: your-secret
      bucketName: your-bucket
  2. ​动态压缩云存储文件​

    从MinIO获取文件流,直接写入ZIP输出流:

    less 复制代码
    @Autowired
    private MinioClient minioClient;
    
    @GetMapping("/download-zip")
    public void downloadZip(@RequestParam List<String> fileNames, HttpServletResponse response) {
        response.setContentType("application/zip");
        response.setHeader("Content-Disposition", "attachment; filename=download.zip");
        
        try (ZipOutputStream zos = new ZipOutputStream(response.getOutputStream())) {
            for (String fileName : fileNames) {
                try (InputStream inputStream = minioClient.getObject(
                        GetObjectArgs.builder().bucket(bucketName).object(fileName).build())) {
                    ZipEntry zipEntry = new ZipEntry(fileName);
                    zos.putNextEntry(zipEntry);
                    IOUtils.copy(inputStream, zos);
                    zos.closeEntry();
                }
            }
        } catch (Exception e) {
            throw new RuntimeException("下载失败", e);
        }
    }

​优化点​​:

  • 直接流式传输,避免内存溢出。
  • 支持断点续传(需结合HTTP Range请求)。

​三、注意事项​

  1. ​内存管理​
    大文件建议使用流式压缩(如ZipOutputStream),避免OutOfMemoryError
  2. ​路径安全​
    使用normalize()过滤非法路径,防止路径遍历攻击。
  3. ​异常处理​
    捕获IOException并返回明确错误状态码(如404文件不存在)。
  4. ​扩展性​
    可结合分片下载(Range请求)支持大文件断点续传。

​四、前端调用示例​

ini 复制代码
// 使用Fetch API触发下载
fetch(`/download-multiple?filenames=file1.txt,file2.jpg`, {
    method: 'GET'
}).then(response => {
    const blob = new Blob([response.blob()]);
    const link = document.createElement('a');
    link.href = URL.createObjectURL(blob);
    link.download = 'files.zip';
    link.click();
});

通过上述方案,可灵活实现本地或云存储的多文件打包下载,根据实际需求选择内存或流式处理。

相关推荐
码农小卡拉1 分钟前
深入解析Spring Boot文件加载顺序与加载方式
java·数据库·spring boot
程序猿追3 分钟前
深度解码昇腾 AI 算力引擎:CANN Runtime 核心架构与技术演进
人工智能·架构
Dragon Wu10 分钟前
Spring Security Oauth2.1 授权码模式实现前后端分离的方案
java·spring boot·后端·spring cloud·springboot·springcloud
晚霞的不甘11 分钟前
CANN 编译器深度解析:TBE 自定义算子开发实战
人工智能·架构·开源·音视频
一个有梦有戏的人31 分钟前
Python3基础:进阶基础,筑牢编程底层能力
后端·python
程序猿追43 分钟前
昇腾算力之锚:深度解读 CANN ascend-toolkit 异构计算架构与工程实践
架构
一枕眠秋雨>o<1 小时前
深入 CANN ops-nn:昇腾 NPU 算子开发的工程化实践与架构哲学
架构
爬山算法1 小时前
Hibernate(88)如何在负载测试中使用Hibernate?
java·后端·hibernate
未来龙皇小蓝1 小时前
RBAC前端架构-01:项目初始化
前端·架构
island13141 小时前
CANN Catlass 算子模板库深度解析:高性能 GEMM 架构、模板元编程与融合算子的显存管理策略
人工智能·神经网络·架构·智能路由器