Retrofit+RxJava已經是目前市場上最主流的網路框架,使用它進行平常的網路請求異常輕鬆,之前也用Retrofit做過上傳檔案和下載檔案,但發現:使用Retrofit做下載預設是不支援進度回撥的,但產品大大要求下載檔案時顯示下載進度,那就不得不深究下了。
接下來我們一起封裝,使用Retrofit+RxJava實現帶進度下載檔案。
github:github.com/shuaijia/Js…
先來看看UML圖:
大家可能還不太清楚具體是怎麼處理的,別急,我們一步步來:
1、添依賴是必須的啦
compile `io.reactivex:rxjava:1.1.0`
compile `io.reactivex:rxandroid:1.1.0`
compile `com.squareup.retrofit2:retrofit:2.0.0-beta4`
compile `com.squareup.retrofit2:converter-gson:2.0.0-beta4`
compile `com.squareup.retrofit2:adapter-rxjava:2.0.0-beta4`
複製程式碼
使用時注意版本號
2、寫回撥
/**
* Description: 下載進度回撥
* Created by jia on 2017/11/30.
* 人之所以能,是相信能
*/
public interface JsDownloadListener {
void onStartDownload();
void onProgress(int progress);
void onFinishDownload();
void onFail(String errorInfo);
}
複製程式碼
這裡就不用多說了,下載的回撥,就至少應該有開始下載、下載進度、下載完成、下載失敗 四個回撥方法。
注意下在onProgress方法中返回進度百分比,在onFail中返回失敗原因。
3、重寫ResponseBody,計算下載百分比
/**
* Description: 帶進度 下載請求體
* Created by jia on 2017/11/30.
* 人之所以能,是相信能
*/
public class JsResponseBody extends ResponseBody {
private ResponseBody responseBody;
private JsDownloadListener downloadListener;
// BufferedSource 是okio庫中的輸入流,這裡就當作inputStream來使用。
private BufferedSource bufferedSource;
public JsResponseBody(ResponseBody responseBody, JsDownloadListener downloadListener) {
this.responseBody = responseBody;
this.downloadListener = downloadListener;
}
@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);
// read() returns the number of bytes read, or -1 if this source is exhausted.
totalBytesRead += bytesRead != -1 ? bytesRead : 0;
Log.e("download", "read: "+ (int) (totalBytesRead * 100 / responseBody.contentLength()));
if (null != downloadListener) {
if (bytesRead != -1) {
downloadListener.onProgress((int) (totalBytesRead * 100 / responseBody.contentLength()));
}
}
return bytesRead;
}
};
}
}
複製程式碼
將網路請求的ResponseBody 和JsDownloadListener 在構造中傳入。
這裡的核心是source方法,返回ForwardingSource物件,其中我們重寫其read方法,在read方法中計算百分比,並將其傳給回撥downloadListener。
4、攔截器
只封裝ResponseBody 是不夠的,關鍵我們需要拿到請求的ResponseBody ,這裡我們就用到了攔截器Interceptor 。
/**
* Description: 帶進度 下載 攔截器
* Created by jia on 2017/11/30.
* 人之所以能,是相信能
*/
public class JsDownloadInterceptor implements Interceptor {
private JsDownloadListener downloadListener;
public JsDownloadInterceptor(JsDownloadListener downloadListener) {
this.downloadListener = downloadListener;
}
@Override
public Response intercept(Chain chain) throws IOException {
Response response = chain.proceed(chain.request());
return response.newBuilder().body(
new JsResponseBody(response.body(), downloadListener)).build();
}
}
複製程式碼
通常情況下攔截器用來新增,移除或者轉換請求或者回應的頭部資訊。
在攔截方法intercept中返回我們剛剛封裝的ResponseBody 。
5、網路請求service
/**
* Description:
* Created by jia on 2017/11/30.
* 人之所以能,是相信能
*/
public interface DownloadService {
@Streaming
@GET
Observable<ResponseBody> download(@Url String url);
}
複製程式碼
注意:
- 這裡@Url是傳入完整的的下載URL;不用擷取
- 使用@Streaming註解方法
6、最後開始請求
/**
1. Description: 下載工具類
2. Created by jia on 2017/11/30.
3. 人之所以能,是相信能
*/
public class DownloadUtils {
private static final String TAG = "DownloadUtils";
private static final int DEFAULT_TIMEOUT = 15;
private Retrofit retrofit;
private JsDownloadListener listener;
private String baseUrl;
private String downloadUrl;
public DownloadUtils(String baseUrl, JsDownloadListener listener) {
this.baseUrl = baseUrl;
this.listener = listener;
JsDownloadInterceptor mInterceptor = new JsDownloadInterceptor(listener);
OkHttpClient httpClient = new OkHttpClient.Builder()
.addInterceptor(mInterceptor)
.retryOnConnectionFailure(true)
.connectTimeout(DEFAULT_TIMEOUT, TimeUnit.SECONDS)
.build();
retrofit = new Retrofit.Builder()
.baseUrl(baseUrl)
.client(httpClient)
.addCallAdapterFactory(RxJavaCallAdapterFactory.create())
.build();
}
/**
* 開始下載
*
* @param url
* @param filePath
* @param subscriber
*/
public void download(@NonNull String url, final String filePath, Subscriber subscriber) {
listener.onStartDownload();
// subscribeOn()改變呼叫它之前程式碼的執行緒
// observeOn()改變呼叫它之後程式碼的執行緒
retrofit.create(DownloadService.class)
.download(url)
.subscribeOn(Schedulers.io())
.unsubscribeOn(Schedulers.io())
.map(new Func1<ResponseBody, InputStream>() {
@Override
public InputStream call(ResponseBody responseBody) {
return responseBody.byteStream();
}
})
.observeOn(Schedulers.computation()) // 用於計算任務
.doOnNext(new Action1<InputStream>() {
@Override
public void call(InputStream inputStream) {
writeFile(inputStream, filePath);
}
})
.observeOn(AndroidSchedulers.mainThread())
.subscribe(subscriber);
}
/**
* 將輸入流寫入檔案
*
* @param inputString
* @param filePath
*/
private void writeFile(InputStream inputString, String filePath) {
File file = new File(filePath);
if (file.exists()) {
file.delete();
}
FileOutputStream fos = null;
try {
fos = new FileOutputStream(file);
byte[] b = new byte[1024];
int len;
while ((len = inputString.read(b)) != -1) {
fos.write(b,0,len);
}
inputString.close();
fos.close();
} catch (FileNotFoundException e) {
listener.onFail("FileNotFoundException");
} catch (IOException e) {
listener.onFail("IOException");
}
}
}
複製程式碼
- 在構造中將下載地址和最後回撥傳入,當然,也可以將儲存地址傳入;
- 在OkHttpClient新增我們自定義的攔截器;
- 注意.addCallAdapterFactory(RxJavaCallAdapterFactory.create()) 支援RxJava;
- 使用RxJava的map方法將responseBody轉為輸入流;
- 在doOnNext中將輸入流寫入檔案;
當然也需要注意下載回撥的各個位置。
獲取更多精彩內容,請關注微信公眾號——Android機動車!