Retrofit+RxJava实现带进度条的文件下载

 更新时间:2019年06月22日 16:52:59   作者:yw_5_24  
这篇文章主要为大家详细介绍了Retrofit+RxJava实现带进度条的文件下载,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

项目中需要使用到更新版本,因此研究了一下Retrofit的下载文件,和进度条效果,其间也遇到了一些坑,写出来加深一下记忆,也为别的同学提供一下思路。

先说一下版本控制吧,通用做法基本上是通过接口获取服务器存储的app版本号,与应用的版本号进行比较,版本较低就去更新,先看一下如何获取应用版本号吧

PackageManager packageManager = mActivity.getPackageManager();

 PackageInfo packageInfo = null;

 try {
 packageInfo = packageManager.getPackageInfo(mActivity.getPackageName(), 0);
 } catch (PackageManager.NameNotFoundException e) {
 e.printStackTrace();
 }

 String versionName = packageInfo.versionName;

可以看到使用的是Context中的getPackageManager方法来获取PackageManager 对象,该对象可用于获取版本的一些信息。

上面的属于附内容,接下来就是关于Retrofit+RxJava实现进度条下载文件的功能,Retrofit本身不提供进度条显示的功能,但Retrofit默认使用Okhttp来进行网络请求,这里就可以自定义拦截器来进行拦截,实现进度。Okhttp的Demo中也为我们提供了一份代码,需要的可以去参考一下Progress.javar,可以看到拦截器的设置:

public class ProgressResponseBody extends ResponseBody {

 private ResponseBody responseBody;

 private ProgressListener progressListener;

 private BufferedSource bufferedSource;


 public ProgressResponseBody(ResponseBody responseBody,ProgressListener progressListener){

 this.responseBody=responseBody;

 this.progressListener=progressListener;
 }


 @Override
 public MediaType contentType() {

 return responseBody.contentType();
 }

 @Override
 public long contentLength() {

 return responseBody.contentLength();
 }

 @Override
 public BufferedSource source() {

 if(bufferedSource==null){

 bufferedSource= Okio.buffer(source(responseBody.source()));
 }

 return bufferedSource;
 }


 private Source source(Source source) {
 return new ForwardingSource(source) {
 long totalBytesRead = 0L;

 @Override
 public long read(Buffer sink, long byteCount) throws IOException {
 //当前读取字节数
 long bytesRead = super.read(sink, byteCount);
 //增加当前读取的字节数,如果读取完成了bytesRead会返回-1
 totalBytesRead += bytesRead != -1 ? bytesRead : 0;
 //回调,如果contentLength()不知道长度,会返回-1

 progressListener.onProgress(totalBytesRead,responseBody.contentLength(),bytesRead,bytesRead==-1);
 return bytesRead;
 }
 };
 }
}

ProgressListener 用来监听进度变化,回调到ProgressInterceptor中,ProgressInterceptor是一个自定义的拦截器,可以看一下代码

public class ProgressInterceptor implements Interceptor {

 @Override
 public Response intercept(Chain chain) throws IOException {

 Response response=chain.proceed(chain.request());
 return response.newBuilder().body(new ProgressResponseBody(response.body(),progressListener)).build();
 }
 static final ProgressListener progressListener=new ProgressListener() {
 @Override
 public void onProgress(long progress, long total, long speed, boolean done) {

 Log.i("log","progress="+progress+"total="+total);
 }
 };
}

为了便于获取progress,可以通过OkHttpClient的addNetworkInterceptor方法直接添加一个自定义的拦截器,例如:

 //为Okhttp设置拦截器
 OkHttpClient client = new OkHttpClient.Builder()
 .addNetworkInterceptor(new Interceptor() {
 @Override public Response intercept(Chain chain) throws IOException {
  Response originalResponse = chain.proceed(chain.request());
  return originalResponse.newBuilder()
  .body(new ProgressResponseBody(originalResponse.body(), progressListener))
  .build();
 }
 })
 .build();

 //进度回调监听
 ProgressListener progressListener=new ProgressListener() {
 @Override
 public void onProgress(long progress, long total, long speed, boolean done) {

 Message message=new Message();

 message.obj=new AmallLoadBean(progress,total);

 progressHandler.sendMessage(message);
 }
 };

这里通过一个创建一个继承自Handler的ProgressHandler静态内部类用于在主线程中刷新进度,顺带提一下,使用static修饰ProgressHandler是因为静态内部类默认不持有外部类对象的引用,需要注意一下Handler的内存泄漏,使用一下写法:

//处理下载版本进度
 public class ProgressHandler extends Handler{

 private WeakReference<Activity> mActivityWeakReference;
 public ProgressHandler(Activity activity){

 mActivityWeakReference=new WeakReference<Activity>(activity);

 }

 @Override
 public void handleMessage(Message msg) {
 if(mActivityWeakReference.get()!=null){
 AmallLoadBean amallLoadBean= (AmallLoadBean) msg.obj;
 long progress=amallLoadBean.getProgress();
 long total=amallLoadBean.getTotal();
 float cp=(float)progress/(float)total;

 }
 }
 }

继续回到下载文件中,我才用的是Retrofit+RxJava的方法来实现,写之前也看了一下别人写的,好像不全,下满也遇到了一些小坑,讲一下吧:

observable.subscribeOn(Schedulers.io())
 .observeOn(Schedulers.io())
 .doOnNext(new Action1<ResponseBody>() {
  @Override
  public void call(ResponseBody responseBody) {

  saveFiles(responseBody);
  }
 })
 .observeOn(AndroidSchedulers.mainThread())
 .subscribe(new Observer<ResponseBody>() {
  @Override
  public void onCompleted() {
  installApk();

  }

  @Override
  public void onError(Throwable e) {

  ToastUtils.getInstance().showToast("请到应用市场下载最新版本");
  }

  @Override
  public void onNext(ResponseBody responseBody) {

  }
 });
 }

通过RxJava的doOnNext在subscribe方法之前存储文件,这里需要注意的是doOnNext方法需要在子线程中执行,调用.observeOn(Schedulers.io())方法,然后再切换到主线程,否则文件下载不下来。当文件下载完成时,在onCompleted方法中执行installApk()方法安装app。需要注意的是这里需要做权限的适配,因为我的是自己封装的因为就不拿出来了,挺简单就自己写吧。保存文件的代码给大家放出来了,通俗的语言:

/**
 * 保存文件
 */
 public void saveFiles(ResponseBody responseBody){

 InputStream inputStream = null;

 FileOutputStream fileOutputStream = null;

 byte[] buffer=new byte[2048];

 int len;

 File file=new File(saveFileName);

 if(!file.exists()){

 file.mkdirs();
 }

 try {
 inputStream=responseBody.byteStream();

 fileOutputStream=new FileOutputStream(file);

 while ((len=inputStream.read(buffer))!=-1){

 fileOutputStream.write(buffer,0,len);
 }

 inputStream.close();

 fileOutputStream.close();

 } catch (Exception e) {
 e.printStackTrace();
 }


 }

在安装文件的时候,需要注意7.0以后的适配,代码看看就好,和拍照适配的原理一直,都是Android对私密性文件的权限问题

 /**
 * 安装apk
 *
 */
 private void installApk() {
 File apkfile = new File(saveFileName);
 if (!apkfile.exists()) {
 return;
 }
 //判断版本号

 if(Build.VERSION.SDK_INT>=Build.VERSION_CODES.N){
 Uri apkUri = FileProvider.getUriForFile(activity, "******.fileprovider", apkfile);
 Intent install = new Intent(Intent.ACTION_VIEW);
 install.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
 //添加这一句表示对目标应用临时授权该Uri所代表的文件
 install.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
 install.setDataAndType(apkUri, "application/vnd.android.package-archive");
 activity.startActivity(install);

 }else{

 Intent i = new Intent(Intent.ACTION_VIEW);
 i.setDataAndType(Uri.parse("file://" + apkfile.toString()), "application/vnd.android.package-archive");
 activity.startActivity(i);
 }

 }

基本上就这些,后续我会在此篇文章上继续补充。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • Spring Boot打war包的实例教程

    Spring Boot打war包的实例教程

    本篇文章主要介绍了Spring Boot打war包的实例教程,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-02-02
  • Java利用POI读取、写入Excel的方法指南

    Java利用POI读取、写入Excel的方法指南

    这篇文章主要给大家介绍了关于Java利用POI读取、写入Excel的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
    2019-03-03
  • 详解Java中的线程模型与线程调度

    详解Java中的线程模型与线程调度

    这篇文章主要介绍了详解Java中的线程模型与线程调度的相关资料,帮助大家更好的理解和使用Java,感兴趣的朋友可以了解下
    2021-02-02
  • MyBatis批量插入的几种方式效率比较

    MyBatis批量插入的几种方式效率比较

    最近工作中遇到了解析excel,然后批量插入,发现这个插入时间比较长,所以想要进行一些优化,下面这篇文章主要给大家介绍了关于MyBatis批量插入的几种方式效率比较的相关资料,需要的朋友可以参考下
    2021-09-09
  • spring mvc利用ajax向controller传递对象的方法示例

    spring mvc利用ajax向controller传递对象的方法示例

    这篇文章主要给大家介绍了关于spring mvc利用ajax向controller传递对象的相关资料,文中通过示例代码将步骤介绍的非常详细,对大家具有一定的参考学习价值,需要的朋友们下面来跟着小编一起学习学习吧。
    2017-07-07
  • JAVA破坏单例模式的方式以及避免方法

    JAVA破坏单例模式的方式以及避免方法

    这篇文章主要介绍了JAVA破坏单例模式的方式以及避免方法,文中讲解非常细致,代码帮助大家更好的理解和学习,感兴趣的朋友可以了解下
    2020-06-06
  • 浅谈JVM内存溢出的几种方式与解决方法

    浅谈JVM内存溢出的几种方式与解决方法

    内存溢出分为两大类:OutOfMemoryError和StackOverflowError,以下举出10个内存溢出的情况,并通过实例代码的方式讲解了是如何出现内存溢出的,感兴趣的可以了解一下
    2024-01-01
  • Windows系统安装JDK小结

    Windows系统安装JDK小结

    这篇文章主要给大家详细介绍了Windows系统安装JDK的方法和步奏,十分的细致,有需要的小伙伴可以参考下
    2016-03-03
  • java中java.util.Date和java.sql.Date之间的转换的示例

    java中java.util.Date和java.sql.Date之间的转换的示例

    java.util.Date是java.sql.Date的父类,有时候在和SqlServer数据库打交道时,也会遇到,本文主要介绍了java中java.util.Date和java.sql.Date之间的转换的示例,具有一定的参考价值,感兴趣的可以了解一下
    2024-05-05
  • Java使用Jedis操作Redis服务器的实例代码

    Java使用Jedis操作Redis服务器的实例代码

    本篇文章主要介绍了Java使用Jedis操作Redis服务器的实例代码,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-08-08

最新评论