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

Spring Boot 3 文件下载、多文件下载以及大文件分片下载、文件流处理、批量操作 和 分片技术

在 Spring Boot 3 中,实现文件下载、多文件下载以及大文件分片下载需要结合以下功能:文件流处理、批量操作 和 分片技术。以下是详细实现方案:

1. 单文件下载

基础的单文件下载实现,可以参考以下代码:

@GetMapping("/download")
public ResponseEntity<Resource> downloadFile(@RequestParam String filename) {
    try {
        Path filePath = Paths.get("/path/to/files").resolve(filename).normalize();
        Resource resource = new UrlResource(filePath.toUri());

        if (!resource.exists()) {
            return ResponseEntity.notFound().build();
        }

        return ResponseEntity.ok()
                .header(HttpHeaders.CONTENT_DISPOSITION, "attachment; filename=\"" + resource.getFilename() + "\"")
                .body(resource);
    } catch (MalformedURLException e) {
        return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR).build();
    }
}

2. 多文件打包下载

对于多文件下载,可以通过将文件打包为 ZIP 文件,再进行下载。

控制器代码

import org.springframework.http.HttpHeaders;
import org.springframework.http.HttpStatus;
import org.springframework.http.ResponseEntity;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.bind.annotation.RestController;

import java.io.*;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.List;
import java.util.zip.ZipEntry;
import java.util.zip.ZipOutputStream;

@RestController
public class MultiFileDownloadController {

    private final String FILE_DIRECTORY = "/path/to/files";

    @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(FILE_DIRECTORY).resolve(filename).normalize();
                if (Files.exists(filePath)) {
                    zos.putNextEntry(new ZipEntry(filename));
                    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.status(HttpStatus.INTERNAL_SERVER_ERROR).build();
        }
    }
}

前端测试请求
使用 fetch 或 axios,发送文件名列表参数,接收 ZIP 文件。

3. 大文件分片下载

对于大文件分片下载,结合 HTTP Range 请求 来实现。

控制器代码

@GetMapping("/download-chunk")
public ResponseEntity<Resource> downloadFileInChunks(
        @RequestParam String filename,
        @RequestHeader(value = "Range", required = false) String rangeHeader) {
    try {
        Path filePath = Paths.get("/path/to/files").resolve(filename).normalize();
        Resource resource = new UrlResource(filePath.toUri());

        if (!resource.exists()) {
            return ResponseEntity.notFound().build();
        }

        long fileSize = Files.size(filePath);
        long rangeStart = 0, rangeEnd = fileSize - 1;

        if (rangeHeader != null && rangeHeader.startsWith("bytes=")) {
            String[] ranges = rangeHeader.replace("bytes=", "").split("-");
            rangeStart = Long.parseLong(ranges[0]);
            if (ranges.length > 1) {
                rangeEnd = Long.parseLong(ranges[1]);
            }
        }

        if (rangeStart > rangeEnd || rangeEnd >= fileSize) {
            return ResponseEntity.status(HttpStatus.REQUESTED_RANGE_NOT_SATISFIABLE).build();
        }

        long chunkSize = rangeEnd - rangeStart + 1;
        InputStream inputStream = Files.newInputStream(filePath);
        inputStream.skip(rangeStart);

        HttpHeaders headers = new HttpHeaders();
        headers.add(HttpHeaders.CONTENT_RANGE, "bytes " + rangeStart + "-" + rangeEnd + "/" + fileSize);
        headers.add(HttpHeaders.ACCEPT_RANGES, "bytes");

        return ResponseEntity.status(HttpStatus.PARTIAL_CONTENT)
                .headers(headers)
                .contentLength(chunkSize)
                .body(new InputStreamResource(inputStream));

    } catch (IOException e) {
        return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR).build();
    }
}

说明

Range 请求:

客户端通过设置 Range: bytes=start-end 来请求文件的部分内容。
响应头包含 Content-Range 和 Accept-Ranges,支持分片下载。

HTTP 状态码:

206 Partial Content:表示成功返回部分内容。
416 Requested Range Not Satisfiable:请求范围无效。

文件流处理:

使用 InputStream.skip() 跳过已读取部分。
确保响应的 Content-Length 与分片大小匹配。

4. 结合前端

分片下载示例(React)

function downloadLargeFile(filename) {
    const chunkSize = 1024 * 1024; // 每片1MB
    let start = 0;

    function fetchChunk() {
        fetch(`/download-chunk?filename=${filename}`, {
            headers: {
                Range: `bytes=${start}-${start + chunkSize - 1}`
            }
        })
            .then(response => {
                if (!response.ok) {
                    throw new Error(`HTTP error! status: ${response.status}`);
                }
                return response.blob();
            })
            .then(blob => {
                saveAs(blob, filename); // 使用 FileSaver.js 保存
                start += chunkSize;
                if (blob.size === chunkSize) {
                    fetchChunk(); // 继续下载下一片
                }
            })
            .catch(err => console.error(err));
    }

    fetchChunk();
}

5. 总结

单文件下载:简单场景适用。
多文件下载:打包为 ZIP 文件传输,适合批量文件场景。
大文件分片下载:支持断点续传,提升用户体验,尤其适合视频流或超大文件传输。
根据具体业务需求,选择合适的实现方案。如果需要更多优化,比如 CDN 加速或并行分片,可以在此基础上扩展实现。


http://www.kler.cn/a/463165.html

相关文章:

  • NCCL源码解读3.1:double binary tree双二叉树构建算法,相比ring环算法的优势
  • JavaScript系列(4)--数值类型专题
  • IDEA试用总结
  • 服务器数据恢复—服务器硬盘亮黄灯的数据恢复案例
  • golang 编程规范 - 项目目录结构
  • 嵌入式学习(21)-正点原子脱机下载器Mini-Pro的使用
  • Java工程师实现视频文件上传minio文件系统存储及网页实现分批加载视频播放
  • 12.30-1-5学习周报
  • 【时时三省】(C语言基础)动态内存函数realloc
  • Node.js 常用命令全攻略
  • VSCode 插件开发实战(十三):如何添加个性化欢迎信息
  • Whiteboard-of-Thought——让大语言模型在白板上写下它们的推理过程,可以大大提高模型在视觉推理能力
  • Github - 如何提交一个带有“verified”标识的commit
  • 【漫话机器学习系列】031.数据增强(Dateset augmentation)
  • 安装、快速入门
  • npm 切换镜像源
  • MySQL5.7主从同步配置
  • mysql查询报错java.sql.SQLException: Illegal mix of collations for operation ‘UNION‘
  • 【智行安全】基于Synaptics SL1680的AI疲劳驾驶检测方案
  • 原生js封装ajax请求以及css实现提示效果和禁止点击效果
  • Android笔试面试题AI答之Android基础(9)
  • 扩充vmware磁盘大小以及分区
  • 细讲前端工程化
  • 使用SDL2搭建简易LVGL模拟器
  • 香港 GPU 服务器托管引领 AI 创新,助力 AI 发展
  • Ubuntu 上高效实现 Texlive 安装和管理