基于OKHttp的大文件下载

1.需求:

根据Url下载大文件,本地创建文件,从输入流读取并写入文件,要求能满足大文件的下载,不能出现OOM

2.引入依赖
xml 复制代码
<!-- OK HTTP -->
<dependency>
    <groupId>com.squareup.okhttp3</groupId>
    <artifactId>okhttp</artifactId>
    <version>4.12.0</version>
</dependency>
<!-- Hutool工具包 -->
<dependency>
    <groupId>cn.hutool</groupId>
    <artifactId>hutool-all</artifactId>
    <version>5.7.16</version>
</dependency>
3.下载文件代码
java 复制代码
import okhttp3.OkHttpClient;
import okhttp3.Request;
import okhttp3.Response;
import cn.hutool.core.io.FileUtil;
import cn.hutool.core.net.URLDecoder;
import java.nio.charset.StandardCharsets;


/**
 * 下载文件并保存
 * @param url 下载地址
 * @param folder 本地保存文件夹路径,比如:E:\test\down
 * @return 文件(下载失败返回null)
 */
public File downloadFile(String url, String folder){
    OkHttpClient client = new OkHttpClient();
    Request request = new Request.Builder().url(url).build();

    try(Response response = client.newCall(request).execute()){
        if(response.isSuccessful()){
            if(Objects.isNull(response.body())){
                logger.info("下载失败,响应体为空");
                return null;
            }
            //从Url提取文件名
            String fileName = getFileNameFromUrl(url);
     
            //拼接完整文件路径
            String filePath = folder + File.separator + fileName;
            //创建本地文件
            File file = new File(filePath);
            FileUtil.touch(file);
            try(
                InputStream in = response.body().byteStream();//获取响应输入流
                FileOutputStream out = new FileOutputStream(file)//创建文件输出流
            ){
                //写入本地文件
                int len;
                byte[] buffer = new byte[4096];
                while((len = in.read(buffer)) != -1) { //从输入流中读取数据到缓冲区
                    out.write(buffer, 0, len); //将缓冲区的数据写入输出流
                }
                out.flush(); //刷新输出流缓冲区
            }
            return file;
        }else{
            logger.info("响应失败:httpStatus={}", response.code());
            return null;
        }
    }catch (Exception e){
        logger.info("文件下载失败,地址:{},错误:{}", url, e.getMessage(), e);
        return null;
    }
}

/**
 * 从Url提取文件名
 * @param url 文件路径
 * @return 文件名
 */
private static String getFileNameFromUrl(String url){
    String uri = URLDecoder.decode(url.trim(), StandardCharsets.UTF_8);
    int index = uri.lastIndexOf("/");
    return  index==-1 ? uri : uri.substring(index+1);
}
相关推荐
CodeAmaz4 分钟前
InnoDB的MVCC机制
java·数据库·mvcc
CoderYanger20 分钟前
贪心算法:4.摆动序列
java·算法·leetcode·贪心算法·1024程序员节
默 语31 分钟前
Spring-AI vs LangChain4J:Java生态的AI框架选型指南
java·人工智能·spring·ai·langchain·langchain4j·spring-ai
kk哥88991 小时前
springboot静态资源的核心映射规则
java·spring boot·后端
老毛肚1 小时前
Java两种代理模式详解
java·开发语言·代理模式
要站在顶端1 小时前
Jenkins PR编号提取&环境变量赋值问题总结
java·servlet·jenkins
愚公移码1 小时前
蓝凌EKP产品:Hibernate 中 SessionFactory、Session 与事务的关系
java·数据库·hibernate·蓝凌
TT哇1 小时前
【每日八股】面经常考
java·面试
何中应1 小时前
【面试题-4】JVM
java·jvm·后端·面试题
老毛肚1 小时前
黑马头条-再回首
java