欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Retrofit+RxJava實現(xiàn)帶進度條的文件下載

 更新時間:2019年06月22日 16:52:59   作者:yw_5_24  
這篇文章主要為大家詳細介紹了Retrofit+RxJava實現(xiàn)帶進度條的文件下載,具有一定的參考價值,感興趣的小伙伴們可以參考一下

項目中需要使用到更新版本,因此研究了一下Retrofit的下載文件,和進度條效果,其間也遇到了一些坑,寫出來加深一下記憶,也為別的同學(xué)提供一下思路。

先說一下版本控制吧,通用做法基本上是通過接口獲取服務(wù)器存儲的app版本號,與應(yīng)用的版本號進行比較,版本較低就去更新,先看一下如何獲取應(yīng)用版本號吧

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 對象,該對象可用于獲取版本的一些信息。

上面的屬于附內(nèi)容,接下來就是關(guān)于Retrofit+RxJava實現(xiàn)進度條下載文件的功能,Retrofit本身不提供進度條顯示的功能,但Retrofit默認使用Okhttp來進行網(wǎng)絡(luò)請求,這里就可以自定義攔截器來進行攔截,實現(xiàn)進度。Okhttp的Demo中也為我們提供了一份代碼,需要的可以去參考一下Progress.javar,可以看到攔截器的設(shè)置:

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 {
 //當(dāng)前讀取字節(jié)數(shù)
 long bytesRead = super.read(sink, byteCount);
 //增加當(dāng)前讀取的字節(jié)數(shù),如果讀取完成了bytesRead會返回-1
 totalBytesRead += bytesRead != -1 ? bytesRead : 0;
 //回調(diào),如果contentLength()不知道長度,會返回-1

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

ProgressListener 用來監(jiān)聽進度變化,回調(diào)到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設(shè)置攔截器
 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();

 //進度回調(diào)監(jiān)聽
 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);
 }
 };

這里通過一個創(chuàng)建一個繼承自Handler的ProgressHandler靜態(tài)內(nèi)部類用于在主線程中刷新進度,順帶提一下,使用static修飾ProgressHandler是因為靜態(tài)內(nèi)部類默認不持有外部類對象的引用,需要注意一下Handler的內(nèi)存泄漏,使用一下寫法:

//處理下載版本進度
 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;

 }
 }
 }

繼續(xù)回到下載文件中,我才用的是Retrofit+RxJava的方法來實現(xiàn),寫之前也看了一下別人寫的,好像不全,下滿也遇到了一些小坑,講一下吧:

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("請到應(yīng)用市場下載最新版本");
  }

  @Override
  public void onNext(ResponseBody responseBody) {

  }
 });
 }

通過RxJava的doOnNext在subscribe方法之前存儲文件,這里需要注意的是doOnNext方法需要在子線程中執(zhí)行,調(diào)用.observeOn(Schedulers.io())方法,然后再切換到主線程,否則文件下載不下來。當(dāng)文件下載完成時,在onCompleted方法中執(zhí)行installApk()方法安裝app。需要注意的是這里需要做權(quán)限的適配,因為我的是自己封裝的因為就不拿出來了,挺簡單就自己寫吧。保存文件的代碼給大家放出來了,通俗的語言:

/**
 * 保存文件
 */
 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對私密性文件的權(quán)限問題

 /**
 * 安裝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);
 //添加這一句表示對目標應(yīng)用臨時授權(quán)該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);
 }

 }

基本上就這些,后續(xù)我會在此篇文章上繼續(xù)補充。

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • Spring Boot打war包的實例教程

    Spring Boot打war包的實例教程

    本篇文章主要介紹了Spring Boot打war包的實例教程,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-02-02
  • Java利用POI讀取、寫入Excel的方法指南

    Java利用POI讀取、寫入Excel的方法指南

    這篇文章主要給大家介紹了關(guān)于Java利用POI讀取、寫入Excel的相關(guān)資料,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-03-03
  • 詳解Java中的線程模型與線程調(diào)度

    詳解Java中的線程模型與線程調(diào)度

    這篇文章主要介紹了詳解Java中的線程模型與線程調(diào)度的相關(guān)資料,幫助大家更好的理解和使用Java,感興趣的朋友可以了解下
    2021-02-02
  • MyBatis批量插入的幾種方式效率比較

    MyBatis批量插入的幾種方式效率比較

    最近工作中遇到了解析excel,然后批量插入,發(fā)現(xiàn)這個插入時間比較長,所以想要進行一些優(yōu)化,下面這篇文章主要給大家介紹了關(guān)于MyBatis批量插入的幾種方式效率比較的相關(guān)資料,需要的朋友可以參考下
    2021-09-09
  • spring mvc利用ajax向controller傳遞對象的方法示例

    spring mvc利用ajax向controller傳遞對象的方法示例

    這篇文章主要給大家介紹了關(guān)于spring mvc利用ajax向controller傳遞對象的相關(guān)資料,文中通過示例代碼將步驟介紹的非常詳細,對大家具有一定的參考學(xué)習(xí)價值,需要的朋友們下面來跟著小編一起學(xué)習(xí)學(xué)習(xí)吧。
    2017-07-07
  • JAVA破壞單例模式的方式以及避免方法

    JAVA破壞單例模式的方式以及避免方法

    這篇文章主要介紹了JAVA破壞單例模式的方式以及避免方法,文中講解非常細致,代碼幫助大家更好的理解和學(xué)習(xí),感興趣的朋友可以了解下
    2020-06-06
  • 淺談JVM內(nèi)存溢出的幾種方式與解決方法

    淺談JVM內(nèi)存溢出的幾種方式與解決方法

    內(nèi)存溢出分為兩大類:OutOfMemoryError和StackOverflowError,以下舉出10個內(nèi)存溢出的情況,并通過實例代碼的方式講解了是如何出現(xiàn)內(nèi)存溢出的,感興趣的可以了解一下
    2024-01-01
  • Windows系統(tǒng)安裝JDK小結(jié)

    Windows系統(tǒng)安裝JDK小結(jié)

    這篇文章主要給大家詳細介紹了Windows系統(tǒng)安裝JDK的方法和步奏,十分的細致,有需要的小伙伴可以參考下
    2016-03-03
  • java中java.util.Date和java.sql.Date之間的轉(zhuǎn)換的示例

    java中java.util.Date和java.sql.Date之間的轉(zhuǎn)換的示例

    java.util.Date是java.sql.Date的父類,有時候在和SqlServer數(shù)據(jù)庫打交道時,也會遇到,本文主要介紹了java中java.util.Date和java.sql.Date之間的轉(zhuǎn)換的示例,具有一定的參考價值,感興趣的可以了解一下
    2024-05-05
  • Java使用Jedis操作Redis服務(wù)器的實例代碼

    Java使用Jedis操作Redis服務(wù)器的實例代碼

    本篇文章主要介紹了Java使用Jedis操作Redis服務(wù)器的實例代碼,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-08-08

最新評論