RxJava+Retrofit+OkHttp深入淺出-終極封裝三(檔案上傳)
背景
在實際運用中上傳是一個必不可少的功能,所以我們在封裝二的基礎上加入上傳的功能,同時需要附帶上傳進度!
效果
實現
1.定義service介面
注意:Multipart是指定大檔案上傳過程中的標示,一般上傳圖片的過程中我們需要附帶資訊,所以我們需要用到@part指定傳遞的數值,MultipartBody.Part是指定傳遞的檔案;
/*上傳檔案*/
@Multipart
@POST("AppYuFaKu/uploadHeadImg")
Observable<BaseResultEntity<UploadResulte>> uploadImage(@Part("uid") RequestBody uid, @Part("auth_key") RequestBody auth_key, @Part MultipartBody.Part file);
- 1
- 2
- 3
- 4
- 1
- 2
- 3
- 4
2.加入進度條
retrofit是基於okhttp的處理,所以我們可以自定義RequestBody,複寫writeTo(BufferedSink sink)方法,得到傳遞的進度資料
public class ProgressRequestBody extends RequestBody {
//實際的待包裝請求體
private final RequestBody requestBody;
//進度回撥介面
private final UploadProgressListener progressListener;
//包裝完成的BufferedSink
private BufferedSink bufferedSink;
public ProgressRequestBody(RequestBody requestBody, UploadProgressListener progressListener) {
this.requestBody = requestBody;
this.progressListener = progressListener;
}
/**
* 重寫呼叫實際的響應體的contentType
* @return MediaType
*/
@Override
public MediaType contentType() {
return requestBody.contentType();
}
/**
* 重寫呼叫實際的響應體的contentLength
* @return contentLength
* @throws IOException 異常
*/
@Override
public long contentLength() throws IOException {
return requestBody.contentLength();
}
/**
* 重寫進行寫入
* @param sink BufferedSink
* @throws IOException 異常
*/
@Override
public void writeTo(BufferedSink sink) throws IOException {
if (null == bufferedSink) {
bufferedSink = Okio.buffer(sink(sink));
}
requestBody.writeTo(bufferedSink);
//必須呼叫flush,否則最後一部分資料可能不會被寫入
bufferedSink.flush();
}
/**
* 寫入,回撥進度介面
* @param sink Sink
* @return Sink
*/
private Sink sink(Sink sink) {
return new ForwardingSink(sink) {
//當前寫入位元組數
long writtenBytesCount = 0L;
//總位元組長度,避免多次呼叫contentLength()方法
long totalBytesCount = 0L;
@Override
public void write(Buffer source, long byteCount) throws IOException {
super.write(source, byteCount);
//增加當前寫入的位元組數
writtenBytesCount += byteCount;
//獲得contentLength的值,後續不再呼叫
if (totalBytesCount == 0) {
totalBytesCount = contentLength();
}
Observable.just(writtenBytesCount).observeOn(AndroidSchedulers.mainThread()).subscribe(new Action1<Long>() {
@Override
public void call(Long aLong) {
progressListener.onProgress(writtenBytesCount, totalBytesCount);
}
});
}
};
}
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
- 41
- 42
- 43
- 44
- 45
- 46
- 47
- 48
- 49
- 50
- 51
- 52
- 53
- 54
- 55
- 56
- 57
- 58
- 59
- 60
- 61
- 62
- 63
- 64
- 65
- 66
- 67
- 68
- 69
- 70
- 71
- 72
- 73
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
- 41
- 42
- 43
- 44
- 45
- 46
- 47
- 48
- 49
- 50
- 51
- 52
- 53
- 54
- 55
- 56
- 57
- 58
- 59
- 60
- 61
- 62
- 63
- 64
- 65
- 66
- 67
- 68
- 69
- 70
- 71
- 72
- 73
3自定義介面,回撥progress進度
public interface UploadProgressListener {
/**
* 上傳進度
* @param currentBytesCount
* @param totalBytesCount
*/
void onProgress(long currentBytesCount, long totalBytesCount);
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
4建立RequestBody物件,加入進度
File file=new File("/storage/emulated/0/Download/11.jpg");
RequestBody requestBody=RequestBody.create(MediaType.parse("image/jpeg"),file);
MultipartBody.Part part= MultipartBody.Part.createFormData("file_name", file.getName(), new ProgressRequestBody(requestBody,
new UploadProgressListener() {
@Override
public void onProgress(long currentBytesCount, long totalBytesCount) {
tvMsg.setText("提示:上傳中");
progressBar.setMax((int) totalBytesCount);
progressBar.setProgress((int) currentBytesCount);
}
}));
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
5.傳遞附帶資訊
和封裝二中post請求的方式一樣,我們需要繼承baseentity,複寫裡面的方法,然後設定需要傳遞的引數,因為是測試介面,所以我的引數直接寫死在entity裡面,part檔案動態指定
/**
* 上傳請求api
* Created by WZG on 2016/10/20.
*/
public class UplaodApi extends BaseEntity {
/*需要上傳的檔案*/
private MultipartBody.Part part;
public UplaodApi(HttpOnNextListener listener, RxAppCompatActivity rxAppCompatActivity) {
super(listener, rxAppCompatActivity);
setShowProgress(true);
}
public MultipartBody.Part getPart() {
return part;
}
public void setPart(MultipartBody.Part part) {
this.part = part;
}
@Override
public Observable getObservable(HttpService methods) {
RequestBody uid= RequestBody.create(MediaType.parse("text/plain"), "4811420");
RequestBody key = RequestBody.create(MediaType.parse("text/plain"), "21f8d9bcc50c6ac1ae1020ce12f5f5a7");
return methods.uploadImage(uid,key,getPart());
}
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
6.post請求處理
請求和封裝二中的請求一樣,通過傳遞一個指定的HttpOnNextListener 物件來回撥來監聽結果資訊,一一對應
private void uploadeDo(){
File file=new File("/storage/emulated/0/Download/11.jpg");
RequestBody requestBody=RequestBody.create(MediaType.parse("image/jpeg"),file);
MultipartBody.Part part= MultipartBody.Part.createFormData("file_name", file.getName(), new ProgressRequestBody(requestBody,
new UploadProgressListener() {
@Override
public void onProgress(long currentBytesCount, long totalBytesCount) {
tvMsg.setText("提示:上傳中");
progressBar.setMax((int) totalBytesCount);
progressBar.setProgress((int) currentBytesCount);
}
}));
UplaodApi uplaodApi = new UplaodApi(httpOnNextListener,this);
uplaodApi.setPart(part);
HttpManager manager = HttpManager.getInstance();
manager.doHttpDeal(uplaodApi);
}
/**
* 上傳回撥
*/
HttpOnNextListener httpOnNextListener=new HttpOnNextListener<UploadResulte>() {
@Override
public void onNext(UploadResulte o) {
tvMsg.setText("成功");
Glide.with(MainActivity.this).load(o.getHeadImgUrl()).skipMemoryCache(true).into(img);
}
@Override
public void onError(Throwable e) {
super.onError(e);
tvMsg.setText("失敗:"+e.toString());
}
};
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
終極封裝專欄
原始碼
建議
相關文章
- RxRetrofit - 終極封裝 - 深入淺出 & 檔案上傳封裝
- RxJava+Retrofit+OkHttp 深入淺出-終極封裝一RxJavaHTTP封裝
- RxJava+Retrofit+OkHttp深入淺出-終極封裝五(資料持久化)RxJavaHTTP封裝持久化
- RxJava+Retrofit+OkHttp深入淺出-終極封裝七(異常處理)RxJavaHTTP封裝
- RxRetrofit - 終極封裝 - 深入淺出 & 斷點續傳封裝斷點
- RxRetrofit - 終極封裝 - 深入淺出 & 異常封裝
- RxRetrofit-終極封裝-深入淺出&入門封裝
- RxRetrofit – 終極封裝 – 深入淺出 & 資料快取封裝快取
- RxRetrofit - 終極封裝 - 深入淺出 & 資料快取封裝快取
- RxRetrofit-終極封裝-深入淺出&網路請求封裝
- RxRetrofit - 終極封裝 - 深入淺出 & 擴充套件 String封裝套件
- RxJava+Retrofit+OkHttp深入淺出-終極封裝六特殊篇(變種String替換Gson自由擴充套件)RxJavaHTTP封裝套件
- RxJava+Retrofit+OkHttp深入淺出-mvp(使用篇)RxJavaHTTPMVP
- PHP實現單檔案、多檔案上傳 封裝 物件導向實現檔案上傳PHP封裝物件
- 淺析檔案上傳漏洞
- YII2檔案上傳驗證,簡單封裝封裝
- [封裝] 小程式直連 oss 上傳檔案JS類封裝JS
- 檔案上傳之三基於flash的檔案上傳
- 深入淺出的webpack構建工具---babel之配置檔案.babelrc(三)WebBabel
- Vue封裝一個簡單輕量的上傳檔案元件Vue封裝元件
- 【Android架構】基於MVP模式的Retrofit2+RXjava封裝之檔案上傳(三)Android架構MVP模式RxJava封裝
- php檔案上傳之多檔案上傳PHP
- Tomcat深入淺出——最終章(六)Tomcat
- 終極套娃 2.0 | 雲原生交付的封裝封裝
- Tomcat深入淺出——Servlet(三)TomcatServlet
- PostgreSQL VACUUM 之深入淺出 (三)SQL
- 深入淺出UML類圖(三)
- 探索網路安全:淺析檔案上傳漏洞
- 什麼是極速檔案傳輸,極速檔案傳輸如何進行大檔案傳輸
- 幾款極好的 JavaScript 檔案上傳外掛JavaScript
- 上傳圖片和檔案出錯!!!
- 單個檔案上傳和批量檔案上傳
- 檔案上傳
- 深入淺出Android BufferQueue-上Android
- VUE封裝的元件庫上傳Vue封裝元件
- SpringMVC 單檔案上傳與多檔案上傳SpringMVC
- WPS檔案保險箱 張三丰的終極法寶
- 深入淺出FE(十四)深入淺出websocketWeb