使用Retrofit下载文件并实现进度监听

1.前言

最近要做一个带进度条下载文件的功能,网上看了一圈,发现好多都是基于OkHttpClient添加拦截器来实现的,个人觉得略显复杂,所以还是采用最简单的方法来实现:基于文件写入来进行进度的监听。

个人觉得不是很科学。只要存在以下两点问题:

  1. 在拦截器中添加进度监听没有真正做到对文件下载的监听,此时,数据在拦截器中应该还只是位于内存中,并没有写到文件中。
  2. 由于是在拦截器中监听,所以有可能出现已经监听到下载完成了,但实际文件写入还没完成,从而导致出现多线程并发问题。
    如有错误,欢迎指正哈~
    所以,决定还是自己撸一下代码,主要还是基于文件写入来进行进度的监听。

2.实现步骤

2.1 设计监听接口

根据需求设计一下接口:

public interface DownloadListener {
   
    void onStart();//下载开始

    void onProgress(int progress);//下载进度

    void onFinish(String path);//下载完成

    void onFail(String errorInfo);//下载失败
}

如果还需下载速度等等,可以自行设计接口和参数。

2.2 编写网络接口Service

public interface DownloadService {
   

    @Streaming
    @GET
    Call<ResponseBody> download(@Url String url);
}

跟正常接口写法基本一致,需要注意的是要添加@Streaming注解。

默认情况下,Retrofit在处理结果前会将服务器端的Response全部读进内存。如果服务器端返回的是一个非常大的文件,则容易发生oom。使用@Streaming的主要作用就是把实时下载的字节就立马写入磁盘,而不用把整个文件读入内存。

2.3 开始网络请求

public class DownloadUtil {
   

    public static void download(String url, final String path, final DownloadListener downloadListener) {
   

        Retrofit retrofit = new Retrofit.Builder()
                .baseUrl("http://www.xxx.com")
                //通过线程池获取一个线程,指定callback在子线程中运行。
                .callbackExecutor(Executors.newSingleThreadExecutor())
                .
  • 3
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 4
    评论
好的,下面是完整的代码示例: ``` public class DownloadManager { private static final int BUFFER_SIZE = 4096; public static void downloadFile(String fileUrl, String saveDir) throws IOException { Retrofit retrofit = new Retrofit.Builder() .baseUrl("http://www.example.com/") .build(); ApiService apiService = retrofit.create(ApiService.class); Call<ResponseBody> call = apiService.downloadFile(fileUrl); Response<ResponseBody> response = call.execute(); if (!response.isSuccessful()) { throw new IOException("Failed to download file: " + response.code()); } long fileSize = response.body().contentLength(); long start = 0; long end = fileSize - 1; String rangeHeader = "bytes=" + start + "-" + end; int bytesRead; byte[] buffer = new byte[BUFFER_SIZE]; long startTime = System.currentTimeMillis(); InputStream inputStream = response.body().byteStream(); File file = new File(saveDir, getFileName(fileUrl)); OutputStream outputStream = new FileOutputStream(file); while ((bytesRead = inputStream.read(buffer)) != -1) { outputStream.write(buffer, 0, bytesRead); end += bytesRead; long endTime = System.currentTimeMillis(); long time = endTime - startTime; long speed = (end - start + 1) / time; long remainingSize = fileSize - (end - start + 1); long remainingTime = remainingSize / speed; long totalTime = time + remainingTime; System.out.println("Downloaded " + (end - start + 1) + " bytes in " + time + "ms, speed: " + speed + "bytes/ms, remaining time: " + remainingTime + "ms, total time: " + totalTime + "ms"); start += bytesRead; rangeHeader = "bytes=" + start + "-" + end; call = apiService.downloadFile(fileUrl, rangeHeader); response = call.execute(); inputStream = response.body().byteStream(); } outputStream.close(); inputStream.close(); System.out.println("File downloaded at: " + file.getAbsolutePath()); } private static String getFileName(String fileUrl) { int index = fileUrl.lastIndexOf("/"); return fileUrl.substring(index + 1); } } ``` 在上述代码中,我们通过 Retrofit 发送了一个 Range 请求头来下载文件的前几个字节,并计算下载速度和剩余时间。然后在循环中,根据上一次下载的结束位置来发送新的 Range 请求头,实现多次下载,最终将整个文件下载完成。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值