java使用文件流上下传,防止内存溢出

背景
最近在做公司的文件服务,遇到有关文件上传下载的内存溢出问题。导致内存溢出的原因是,上传下载时,将文件流转成了字节数组,导致程序大量使用内存。后来改成文件流方式避免了内存的溢出。

接口实现

上传接口:

    @RequestMapping("uploadFile")
    public Boolean uploadFile(@RequestParam("file") MultipartFile file) {
		String rootPath = System.getProperty("user.dir");
        String filePath = rootPath + "/files/test.mp4";
        try {
            File newFile = new File(filePath);
            if (!newFile.exists()) {
                newFile.getParentFile().mkdirs();
                newFile.createNewFile();
            }
            // 会覆盖原文件
            file.transferTo(newFile);
            return true;
        } catch (IOException e) {
            e.printStackTrace();
            return false;
        }
	}

下载接口:


    @RequestMapping("downloadFile")
    public void downloadFile(HttpServletRequest request, HttpServletResponse response) {

		String rootPath = System.getProperty("user.dir");
        String filePath = rootPath + "/files/test.mp4";
        
        File file = new File(filePath);
        InputStream is = null;
        if (file.exists()) {
            is = new FileInputStream(file);
        }
        
        if (is != null) {

            String fileName = filePath.substring(filePath.lastIndexOf('/') + 1);
			// 设置下载文件名
            String userAgent = request.getHeader("User-Agent");
            userAgent = userAgent == null ? "" : userAgent.toLowerCase();
            // 针对IE或者以IE为内核的浏览器:
            if (userAgent.contains("msie") || userAgent.contains("trident")) {
                fileName = URLEncoder.encode(fileName, "UTF-8");
                fileName = fileName.replace("+", " ");
            } else {
                fileName = new String(fileName.getBytes(), "iso-8859-1");
            }
            
            response.reset();
            response.setContentType("application/octet-stream; charset=UTF-8");
            response.setHeader("Content-Disposition", "attachment;filename=\"" + fileName + "\"");

            long length = is.transferTo(response.getOutputStream());
            response.addHeader("Content-Length", "" + length);
            is.close();
        } else {
            response.setContentType("text/plain;charset=UTF-8");
            response.getWriter().write("未能找到资源文件");
            response.setStatus(404);
            return;
        }
    }

你可能感兴趣的文章:

  • 1
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
您好!如果在使用JavaFileChannel下载文件时出现内存溢出的问题,可能是因为您一次性尝试读取或写入过大的文件块导致的。以下是一些解决方法: 1. 使用ByteBuffer进行分块读取:通过创建一个较小的ByteBuffer对象,例如4KB或8KB,并使用循环读取文件的一部分,然后处理该部分数据。这样可以避免一次性将整个文件加载到内存中。 ```java FileChannel channel = new FileInputStream(sourceFile).getChannel(); ByteBuffer buffer = ByteBuffer.allocate(8 * 1024); // 8KB buffer while (channel.read(buffer) != -1) { buffer.flip(); // 处理buffer中的数据 buffer.clear(); } ``` 2. 使用内存映射文件(Memory-Mapped File):内存映射文件允许您将文件直接映射到内存中,从而可以像访问内存一样访问文件内容。这样可以避免将整个文件加载到内存中。 ```java FileChannel channel = new RandomAccessFile(sourceFile, "r").getChannel(); MappedByteBuffer buffer = channel.map(FileChannel.MapMode.READ_ONLY, 0, channel.size()); // 处理buffer中的数据 ``` 3. 降低缓冲区大小:在使用FileChannel时,可以通过减小ByteBuffer的大小来降低内存消耗。较小的缓冲区大小可能会导致性能损失,但可以避免内存溢出。 ```java FileChannel channel = new FileInputStream(sourceFile).getChannel(); ByteBuffer buffer = ByteBuffer.allocate(1 * 1024); // 1KB buffer while (channel.read(buffer) != -1) { buffer.flip(); // 处理buffer中的数据 buffer.clear(); } ``` 请根据您的具体需求选择合适的解决方法。希望对您有所帮助!如果您有任何其他问题,请随时提问。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值