Retrofit 2.0 超能实践(四),完成大文件断点下载
作者:码小白
文/CSDN 博客
本文出自:http://blog.csdn.net/sk719887916/article/details/51988507 码小白
通过前几篇系统的介绍和综合运用,忘记介绍文件下载功能了,有朋友问到,目前APP文件下载主要有断点续传,多线程并发下载,多类型下载,今天就介绍下其Retrofit下载文件功能。
Retrofit 2.0
超能实践,完美支持Https传输Retrofit2.0
完美同步Cookie实现免登录Retrofit 2.0 超能实践(三),轻松实现文件/图片上传
基于Retrofit2.0 封装的超好用的RetrofitClient工具类
玩转IOC,教你徒手实现自定义的Retrofit框架
ApiService
编写API,执行下载接口功能。
public interface ApiService {@Streaming@GETObservable<ResponseBody> downloadFile(@Url String fileUrl);
}
url由于是可变的,因此用 @URL
注解符号来进行指定,大文件官方建议用 @Streaming
来进行注解,不然会出现IO异常,小文件可以忽略不注入。如果想进行断点续传的话 可以在此加入header,但不建议直接在api中写死,每个下载的请求大小是不同的,在拦截器加入更为妥善。
DownLoadManager
实现一个下载管理者 来进行文件写入,类型判断等,如果想做完善,可以判断下http的请求头 content-length
, content- type
, RANGE
第一个用来判断文件大小,第二个文件类型,第三个是文件从哪儿开始下载,如果对下载来源校验可以加入referer
, 不是目标来源的可以不予下载权限。
public class DownLoadManager {private static final String TAG = "DownLoadManager";private static String APK_CONTENTTYPE = "application/vnd.android.package-archive";private static String PNG_CONTENTTYPE = "image/png";private static String JPG_CONTENTTYPE = "image/jpg";private static String fileSuffix="";public static boolean writeResponseBodyToDisk(Context context, ResponseBody body) {Log.d(TAG, "contentType:>>>>"+ body.contentType().toString());String type = body.contentType().toString();if (type.equals(APK_CONTENTTYPE)) {fileSuffix = ".apk";} else if (type.equals(PNG_CONTENTTYPE)) {fileSuffix = ".png";}// 其他同上 自己判断加入String path = context.getExternalFilesDir(null) + File.separator + System.currentTimeMillis() + fileSuffix;Log.d(TAG, "path:>>>>"+ path);try {// todo change the file location/name according to your needsFile futureStudioIconFile = new File(path);InputStream inputStream = null;OutputStream outputStream = null;try {byte[] fileReader = new byte[4096];long fileSize = body.contentLength();long fileSizeDownloaded = 0;inputStream = body.byteStream();outputStream = new FileOutputStream(futureStudioIconFile);while (true) {int read = inputStream.read(fileReader);if (read == -1) {break;}outputStream.write(fileReader, 0, read);fileSizeDownloaded += read;Log.d(TAG, "file download: " + fileSizeDownloaded + " of " + fileSize);}outputStream.flush();return true;} catch (IOException e) {return false;} finally {if (inputStream != null) {inputStream.close();}if (outputStream != null) {outputStream.close();}}} catch (IOException e) {return false;}
}
}
上面只是简单的对不同类型文件进行了判断,其他类型参考http
协议描述类型自行加入,断点续传目前暂时没实现,基于HttpClinet下载的介绍很多,关键思想是记录一条下载产生后的RANGE
到数据库里,每条下载行对应一条数据,下载ID是区分唯一的主键, 包含filename
, fileSize, fileType, downTime, downRange
status
基本可满足多线程下载数据要求,在Retrofit结合RXJava后 解决了线程安全问题,实现多线程下载就更so yi z, 这里我也不做多的介绍。
#调用下载
OkHttpClient okHttpClient = new OkHttpClient.Builder().build();Retrofit retrofit = new Retrofit.Builder().client(okHttpClient).baseUrl(url).build();
Api Service apiService = retrofit.create(ApiService.class);apiService..download(url1,new Subscriber<ResponseBody>() {@Overridepublic void onCompleted() {}@Overridepublic void onError(Throwable e) {}@Overridepublic void onNext(ResponseBody responseBody) {if (DownLoadManager.writeResponseBodyToDisk(MainActivity.this, responseBody)) {Toast.makeText(MainActivity.this, "Download is sucess", Toast.LENGTH_LONG).show();}}});}});
当 然我们可以在header中加入上次下载的进度大小,就可以实现基本的断点下载了,多线程下载目前业内主要由下面思想实现,我们可以请求到文件总长度的时候,均分三段大小区间,来开启三个新线程去下载,下载完后再以唯一的文件ID,将三段文件以此追加到一个文件就可以了,当然在retrofit结合RxJava,你无需开启新线程,只需订阅一下 Subscriber到IO线程即可,这里注意的是你需要判断最后的文件大小和HashCode来进行验证文件完整性,不然视频文件丢包了没多大区别,最多会卡帧,但是apk丢包了却无法解析安装了。
断点下载姿势开启!
#断点下载
##apiServiece
@GET
@Streaming
Observable<Response<ResponseBody>> download(@Header("Range") String range, @Url String url);
##API
DownLoadInfos里面包含name,lenth,.url,等,这里不再贴了。直接看API
DownloadType 里面记录要有stats,mLastModify,downloded ,savepath等
public Observable<DownloadInfos> download(@NonNull final String url, @NonNull final String saveName,@Nullable final String savePath) {
return downloadDispatcher(url, saveName, savePath);
}private Observable<DownloadStatus> downloadDispatcher(final String url, final String saveName, final String savePath) {return getDownloadType(url).flatMap(new Func1<DownloadType, Observable<DownloadStatus>>() {@Overridepublic Observable<DownloadInfos> call(DownloadType type) {try {type.prepare();} catch (IOException | ParseException e) {return Observable.error(e);}try {return type.start();} catch (IOException e) {return Observable.error(e);}}}).doOnCompleted(new Action0() {@Overridepublic void call() {mDownloadManager.delete(url);}}).doOnError(new Action1<Throwable>() {@Overridepublic void call(Throwable throwable) {mDownloadManager.delete(url);}}).doOnUnsubscribe(new Action0() {@Overridepublic void call() {mDownloadManager.delete(url)});
}
开始下载:
@OverrideObservable<DownloadInfos> start() {Log.i(TAG, "Normal download start!!");return mDownloadHelper.getDownloadApi().download(null, mUrl).subscribeOn(Schedulers.io()).flatMap(new Func1<Response<ResponseBody>, Observable<DownloadStatus>>() {@Overridepublic Observable<DownloadStatus> call(final Response<ResponseBody> response) {return normalSave(response);}}).onBackpressureLatest().retry(new Func2<Integer, Throwable, Boolean>() {@Overridepublic Boolean call(Integer integer, Throwable throwable) {return mDownloadHelper.retry(integer, throwable);}});}
##调用
DownloadAgenti.getInstance(mContext).observeOn(Schedulers.io()).observeOn(AndroidSchedulers.mainThread()).subscribe(new Subscriber<DownloadInfos>() {@Overridepublic void onCompleted() {// todo}@Overridepublic void onError(Throwable e) {todo}@Overridepublic void onNext(final DownloadStatus status) {todo}});
#总结
步骤:
请求文件总大小
根据机型高低,分配多个线程下载
记录下载进度,大小,类型等到数据库
同时更新UI和通知栏,提示用户
下载结束后更新数据库下载数据,追加组合文件
判断文件大小,检验文件大小
下载功能是每个应用必须的功能,因此下载还是比较重要的一个模块,好的下载架构 会考虑到设备的CPU大小,网络状态,以及外部sdcard的大小,动态进行性能,流量分配,方便用户更好的体验和使用你的APP.
具体可以阅读 - 基于Retrofit2.0 封装的超好用的RetrofitClient工具类 中的写法
项目github:https://github.com/Tamicer/FastDownloader
作者:Tamic 更多原创关注开发者技术前线
Retrofit 2.0 超能实践(四),完成大文件断点下载相关推荐
- Retrofit 2.0 超能实践(三),轻松实现多文件/图片上传/Json字符串/表单
通过前两篇姿势的入门 Retrofit 2.0 超能实践,完美支持Https传输 Retrofit2.0 完美同步Cookie实现免登录 本文出自:http://blog.csdn.net/sk719 ...
- Retrofit 2.0 超能实践(一),okHttp完美支持Https传输
http: //blog.csdn.net/sk719887916/article/details/51597816 Tamic首发 前阵子看到圈子里Retrofit 2.0,RxJava(Andro ...
- Python组织文件 实践:查找大文件、 用Mb、kb显示文件尺寸 、计算程序运行时间...
这个小程序很简单原本没有记录下来的必要,但在编写过程中又让我学到了一些新的知识,并且遇到了一些不能解决的问题,然后,然后就很有必要记录一下. 这个程序的关键是获取文件大小,本来用 os.path.ge ...
- Python组织文件 实践:查找大文件、 用Mb、kb显示文件尺寸 、计算程序运行时间
这个小程序很简单原本没有记录下来的必要,但在编写过程中又让我学到了一些新的知识,并且遇到了一些不能解决的问题,然后,然后就很有必要记录一下. 这个程序的关键是获取文件大小,本来用 os.path.ge ...
- java写入excel文件内存不足,java 导出 excel 最佳实践,java 大文件 excel 避免OOM(内存溢出) excel 工具框架...
产品需求 产品经理需要导出一个页面的所有的信息到 EXCEL 文件. 需求分析 对于 excel 导出,是一个很常见的需求. 最常见的解决方案就是使用 poi 直接同步导出一个 excel 文件. 客 ...
- 超详细的java生成excel文件并下载
在网上随手一搜,可以搜到很多java生成excel文件相关的博客,但每个都有不同,核心点说清楚了,但具体运用的时候,相信大家或多或少都没法一次直接运用,这样每次去找并且运用的时候很费时间,所以这也是我 ...
- Retrofit 2.0 轻松实现多文件/图片上传/Json字符串/表单
如果嫌麻烦直接可以用我封装好的库:Novate: https://github.com/Tamicer/Novate 通过对Retrofit2.0的前两篇的基础入门和案例实践,掌握了怎么样使用Retr ...
- iis 6.0 中一些文件无法下载问题解决
今天把一个客户网站从iis5.0转到iis6.0服务器上后,发现一些文件无法下载,下载时提示"HTTP 错误 404 - 文件或目录未找到",在网上搜索了一下发现是由于MIME设置 ...
- 随机森林的特征 是放回抽样么_机器学习超详细实践攻略(10):随机森林算法详解及小白都能看懂的调参指南...
一.什么是随机森林 前面我们已经介绍了决策树的基本原理和使用.但是决策树有一个很大的缺陷:因为决策树会非常细致地划分样本,如果决策树分得太多细致,会导致其在训练集上出现过拟合,而如果决策树粗略地划分样 ...
最新文章
- java调用asp.net webapi_通过HttpClient 调用ASP.NET Web API示例
- MySQL内核调试_MySQL内核技术之“Opt_trace_系列”
- 重庆计算机二本专业有哪些专业,重庆高考计算机类分数线
- FreeMarker快速上手
- bzoj 2763 [JLOI2011]飞行路线——分层图
- 3D Game Programming with directx 11 习题答案 8.3
- 不一样的结果,不一样的人生
- RabbitMQ学习之集群镜像模式配置
- 11.05T5 另类背包
- 论文总结Graph Neural Networks-A review of methods and Applications
- 自建 bitwarden 密码管理服务
- 去水印小程序源码,全新界面无加密,平台支持微信小程序和QQ小程序。支持解析抖音、快手、皮皮虾和微视等平台。带PHP下载接口。支持微信QQ流量主
- 员工满意度调查问卷的设计注意事项
- [BZOJ]4453: cys就是要拿英魂! 单调栈+二分+hash
- Windows注册表命令(最简单明了)
- 从1到无穷大—机器学习篇
- mysql vtype_ExtJs6学习笔记 -- 自定义 vtype
- 【Linux】虚拟机VMware的Ubuntu使用vi指令的方向键和backspace空格键乱码
- mnist数据集百度云链接
- 区块链入门系列之梅克尔帕特里夏树
热门文章
- HTML5 Canvas制作雷达图实战
- Nachos Lab4 文件系统
- pygame五子棋人机游戏
- 西南科技大学OJ题 顺序表插入操作的实现0943
- 4.7 Python设置代码格式
- 有效的数独-LeetCode20
- Android源码编译 HTC One/M7中文触摸CWM6.0.3.7版Recovery,专为白卡机(S-OFF)量身打造,再也不用担心线刷不了Recovery了
- 西游之路——python全栈——Django之ORM操作
- C++ — 类型萃取
- 手机相机变成QQ摄像头 先锋P80W