国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 系統 > Android > 正文

RxJava+Retrofit+OkHttp實現文件上傳

2019-10-22 18:23:00
字體:
來源:轉載
供稿:網友

背景

在實際運用中上傳是一個必不可少的功能,所以我們在封裝二的基礎上加入上傳的功能,同時需要附帶上傳進度!

效果

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);

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);          }        });      }    };  }}

3自定義接口,回調progress進度

public interface UploadProgressListener {  /**   * 上傳進度   * @param currentBytesCount   * @param totalBytesCount   */  void onProgress(long currentBytesCount, long totalBytesCount);}

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);     }   }));

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());  }}

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());    }  };

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持VEVB武林網。


注:相關教程知識閱讀請移步到Android開發頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 九寨沟县| 许昌市| 兰州市| 娱乐| 漠河县| 剑河县| 青阳县| 祥云县| 沽源县| 如东县| 延吉市| 大竹县| 额尔古纳市| 全州县| 萨嘎县| 镇远县| 临澧县| 北碚区| 南华县| 大姚县| 于都县| 威宁| 韶关市| 河津市| 桦甸市| 柳州市| 五台县| 剑川县| 高陵县| 西乌| 行唐县| 阳东县| 邵东县| 崇信县| 洪江市| 太谷县| 汉寿县| 柞水县| 定安县| 杭锦旗| 济南市|