Retrofit2.0實(shí)現(xiàn)文件批量上傳監(jiān)聽(tīng)進(jìn)度的一種蠢辦法

最近項(xiàng)目需要做文件批量上傳的進(jìn)度監(jiān)聽(tīng),這也就是一個(gè)常見(jiàn)的需求。但是項(xiàng)目中用的是Retrofit,官方并沒(méi)有提供此類的API,于是只能Google啦。找了一圈,資源很少,僅僅找到了幾篇單文件上傳的進(jìn)度監(jiān)聽(tīng),不能直接ctrl+c ctrl+v啦,只有自己用笨辦法,稍微封裝一下。

PS:本人習(xí)慣在代碼中分析,總結(jié)性的文字很少。

由于沒(méi)有測(cè)試接口,于是就使用項(xiàng)目中的上傳接口。先貼一張最終實(shí)現(xiàn)的效果圖。

湊活看看吧

以上是公司的項(xiàng)目錄的gif,錄的不太流暢,信息都是事先填好的,湊活看看吧。由于是內(nèi)部使用的app,所以界面是真的丑啊。。。

廢話說(shuō)太多了,正式開(kāi)始吧。

批量上傳


從效果圖可以看到,客戶簽字完畢后,取車成功就是上傳操作了。需求是要將之前所有信息全部上傳到服務(wù)器,包含了文字和大量的圖片(所有字段都填滿,最多可以達(dá)到上百?gòu)垼jP(guān)于Retrofit的批量上傳,我使用的是Multipart,具體使用請(qǐng)自行百度Google,這里簡(jiǎn)要貼一下代碼。

先貼出Api接口代碼,info字段為將所有文字參數(shù)封裝成的json串上傳:

//上傳取車信息
    @Multipart
    @POST("reportgetcarlisttoservice.tag")
    Observable<CheckBaseBean> uploadGetCarData(@Part("info") RequestBody info,                                           
                                               @PartMap Map<String, RequestBody> imgs);

(上傳部分代碼較長(zhǎng),篩選了關(guān)鍵部分的代碼貼出來(lái))

public void upload() {
      //檢測(cè)上傳參數(shù)完整性  略
      ...

      //開(kāi)啟一個(gè)線程   由于涉及大量圖片的信息,壓縮等耗時(shí)操作,開(kāi)啟一個(gè)線程處理是必須的
      new Thread(() -> {
            //文字參數(shù)
            UploadGetCarDataRequest request = new UploadGetCarDataRequest();
            request.orderkey = Session.currentOrderKey;
            if (Integer.parseInt(Session.currentOrderStatus) >= 33 && Integer.parseInt(Session.currentOrderStatus) != 34) {
                request.orderstate = Integer.parseInt(Session.currentOrderStatus);
            } else {
                request.orderstate = 33;
            }
            request.sgwxsm = spotData.getWeixiushuoming();
            request.fsgwxsm = spotData.getFeiweixiushuoming();
            ...
            request.remark = otherData.getBeizhu();
            request.deleteimages = updateGetCarInfo.getDelete_imgs();

            //使用Map存儲(chǔ)RequestBody,打包上傳圖片
            Map<String, RequestBody> bodyMap = new HashMap<>();

            //判斷圖片若以http開(kāi)頭,則表示服務(wù)器圖片(已經(jīng)上傳過(guò)),則不必上傳,反之本地圖片壓縮后上傳
            if (!spotData.getMenpai().startsWith("http")) {
                if (!new File(spotData.getMenpai()).exists()) {
                    Message m = Message.obtain();
                    m.what = 3;
                    m.obj = "接車門牌地址照片有問(wèn)題,請(qǐng)檢查";
                    mHandler.sendMessage(m);
                    return;
                }
                //checkFile方法為檢驗(yàn)圖片大小,若大于200k,則壓縮到200k以下,節(jié)省上傳流量和時(shí)間
                File menpai = checkFile(new File(spotData.getMenpai()));
                bodyMap.put("0" + "-0-0\";filename=\"" + menpai.getName(), new UploadFileRequestBody(RequestBody.create(MediaType.parse("image/png"), menpai), mProgressListener, Constant.GET_MEN_PAI + ""));
                totalLength += menpai.length();
            }
            ...

              mSubscription = GoldKeyRetrofit.getDefaultRetrofit(mContext)
                    .create(GoldKeyService.class)
                     //文字參數(shù)轉(zhuǎn)換成json串上傳,RequestFactory是我自己封裝的將Request轉(zhuǎn)換為json的類
                    .uploadGetCarData(RequestBody.create(MediaType.parse("application/json"), RequestFactory.getInstance().getParams(request)), bodyMap)
                    .subscribeOn(Schedulers.io())
                    .observeOn(AndroidSchedulers.mainThread())
                    .subscribe(new Observer<CheckBaseBean>() {
                        @Override
                        public void onCompleted() {
                        }

                        @Override
                        public void onError(Throwable e) {
                            mView.showToast("上傳失敗,請(qǐng)檢查網(wǎng)絡(luò)是否通暢");
                            mView.showUploadProgress(0, ProgressView.STATE_FAILURE);
                            isFinish = true;
                        }

                        @Override
                        public void onNext(CheckBaseBean checkBaseBean) {
                            if (checkBaseBean.sign) {
                                mView.showToast("上傳成功");
                                mView.showUploadProgress(100, ProgressView.STATE_SUCCESS);
                                mHandler.sendEmptyMessageDelayed(2, 2000);
                                isFinish = true;
                                clearDB();
                                clearCache();
//                                mView.backToMain();
                            } else {
                                mView.showUploadProgress(0, ProgressView.STATE_FAILURE);
                                mView.showToast("上傳失敗,請(qǐng)反饋給開(kāi)發(fā)人員,謝謝");
                                isFinish = true;
                            }
                        }
                    });
        }).start();
      }
}

批量上傳就先說(shuō)到這里,其實(shí)講了等于沒(méi)講,不懂的依然是不懂,哈哈。

上傳進(jìn)度監(jiān)聽(tīng)


仔細(xì)說(shuō)一下這一塊。
以前做上傳的時(shí)候,用的是Xutils框架,其提供了上傳和下載的進(jìn)度監(jiān)聽(tīng),而強(qiáng)大的Retrofit居然沒(méi)有提供相關(guān)的Api,好蛋疼。
百度了一下相關(guān)的資料,發(fā)現(xiàn)大多數(shù)都是模仿Retrofit官方提供的ChunkingConverter,寫一個(gè)轉(zhuǎn)換器來(lái)監(jiān)聽(tīng)進(jìn)度。剛開(kāi)始我也是采用這種思路,想通過(guò)封裝一個(gè)Converter來(lái)監(jiān)聽(tīng)多文件上傳的進(jìn)度。但是開(kāi)發(fā)過(guò)程中碰到了幾個(gè)坑。首先,添加轉(zhuǎn)換器是通過(guò)Retrofit.Builder創(chuàng)建Retrofit實(shí)例時(shí)添加的,而這個(gè)實(shí)例我們通常使用的是單例,其他接口又沒(méi)必要添加這個(gè)轉(zhuǎn)換器,所以要使用上傳監(jiān)聽(tīng)必須重新new一個(gè)Retrofit實(shí)例,很麻煩。第二,Converter只能拿到單個(gè)RequestBody的數(shù)據(jù),但是要實(shí)現(xiàn)多文件的監(jiān)聽(tīng),很麻煩。第三,大姨夫來(lái)了,很煩。
于是換個(gè)思路,既然converter是通過(guò)監(jiān)聽(tīng)RequestBody獲取其已寫的字節(jié),那么我們?yōu)槭裁床恢苯臃庋b一個(gè)RequsetBody,直接返回這些數(shù)據(jù)呢?
首先,先定義一個(gè)回調(diào)接口:

public interface ProgressListener {
    //要是單文件上傳,就不必再根據(jù)字節(jié)去計(jì)算了,直接在requestbody中計(jì)算好進(jìn)度直接返回
    void onProgress(int progress, String tag);
    //處理多文件時(shí),需要獲取每個(gè)文件的即時(shí)上傳量來(lái)計(jì)算整體的進(jìn)度
    void onDetailProgress(long written, long total, String tag);
}

先貼出自己封裝的UploadFileRequestBody:

public class UploadFileRequestBody extends RequestBody {

    private RequestBody mRequestBody;
    private ProgressListener mProgressListener;

    private BufferedSink bufferedSink;

    //每個(gè)RequestBody對(duì)應(yīng)一個(gè)tag,存放在map中,保證計(jì)算的時(shí)候不會(huì)出現(xiàn)重復(fù)
    private String tag;

    public UploadFileRequestBody(File file, ProgressListener progressListener, String tag) {
        this.mRequestBody = RequestBody.create(MediaType.parse("multipart/form-data"), file);
        this.mProgressListener = progressListener;
        this.tag = tag;
    }

    //其實(shí)只是添加一個(gè)回調(diào)和tag標(biāo)識(shí),實(shí)際起作用的還是requestBody
    public UploadFileRequestBody(RequestBody requestBody, ProgressListener progressListener, String tag) {
        this.mRequestBody = requestBody;
        this.mProgressListener = progressListener;
        this.tag = tag;
    }

    //返回了requestBody的類型,想什么form-data/MP3/MP4/png等等等格式
    @Override
    public MediaType contentType() {
        return mRequestBody.contentType();
    }

    //返回了本RequestBody的長(zhǎng)度,也就是上傳的totalLength
    @Override
    public long contentLength() throws IOException {
        return mRequestBody.contentLength();
    }

    @Override
    public void writeTo(BufferedSink sink) throws IOException {
        if (bufferedSink == null) {
            //包裝
            bufferedSink = Okio.buffer(sink(sink));
        }
        //寫入
        mRequestBody.writeTo(bufferedSink);
        //必須調(diào)用flush,否則最后一部分?jǐn)?shù)據(jù)可能不會(huì)被寫入
        bufferedSink.flush();
    }

    private Sink sink(Sink sink) {
        return new ForwardingSink(sink) {
            //當(dāng)前寫入字節(jié)數(shù)
            long bytesWritten = 0L;
            //總字節(jié)長(zhǎng)度,避免多次調(diào)用contentLength()方法
            long contentLength = 0L;

            @Override
            public void write(Buffer source, long byteCount) throws IOException {
                super.write(source, byteCount);
                if (contentLength == 0) {
                    //獲得contentLength的值,后續(xù)不再調(diào)用
                    contentLength = contentLength();
                }
                //增加當(dāng)前寫入的字節(jié)數(shù)
                bytesWritten += byteCount;
                //回調(diào)上傳接口
                mProgressListener.onProgress((int) ((double) bytesWritten / (double) contentLength) * 100, tag);
                mProgressListener.onDetailProgress(bytesWritten, contentLength, tag);
            }
        };
    }
}

在上傳過(guò)程中,我們可以通過(guò)ProgressListener接口實(shí)時(shí)拿到每個(gè)RequestBody上傳的字節(jié)數(shù)。我們的需求是計(jì)算出所有文件上傳的總進(jìn)度。其實(shí)就是要計(jì)算出 (所有文件已上傳的大小)/(所有文件的累加大小)。分母上文件總大小我們可以在創(chuàng)建RequestBody時(shí),使用一個(gè)long變量,將每個(gè)file.length()累加,即可得到。分子上的已上傳大小是要由回調(diào)中的bytesWritten參數(shù)統(tǒng)計(jì)而得。我們使用一個(gè)Map來(lái)記錄每個(gè)文件的上傳大小,通過(guò)標(biāo)識(shí)tag來(lái)區(qū)分每個(gè)文件:

private Map<String, Long> mProgresses2 = new HashMap<>();

private void upload(){
      ...
      File menpai = checkFile(new File(spotData.getMenpai()));
      //創(chuàng)建UploadFileRequestBody對(duì)象,傳入tag(保證不同)
      bodyMap.put("0" + "-0-0\";filename=\"" + menpai.getName(), new UploadFileRequestBody(RequestBody.create(MediaType.parse("image/png"), menpai), mProgressListener, Constant.GET_MEN_PAI + ""));
      //totalLength記錄文件總大小 (記得在每次執(zhí)行上傳時(shí),重置為0L)
      totalLength += menpai.length();
      ...
}

由于進(jìn)度回調(diào)處理方式是可以統(tǒng)一處理的,所以所有的RequestBody都使用同一個(gè)mProgressListener:

mProgressListener = new ProgressListener() {
            @Override
            public void onProgress(int progress, String tag) {
            }

            @Override
            public void onDetailProgress(long written, long total, String tag) {
                //回調(diào)做的唯一事情就是實(shí)時(shí)更新這個(gè)Map
                mProgresses2.put(tag, written);
            }
        };

我們遍歷整個(gè)map,累加所有的value值,便是當(dāng)前所有的上傳大小了,即拿到了最終要得到的上傳進(jìn)度。接下來(lái)要做的就是更新UI,顯示這個(gè)進(jìn)度了,我們可以開(kāi)啟一個(gè)線程循環(huán)更新,也可以通過(guò)handler。我這里采用的是handler:

private Handler mHandler = new Handler() {
        @Override
        public void handleMessage(Message msg) {
            //isFinish是一個(gè)非常關(guān)鍵的標(biāo)記位,記錄是否還需要發(fā)送消息(在取消上傳,或者上傳成功失敗后,置為true),若沒(méi)有這個(gè)標(biāo)記位,將在后臺(tái)無(wú)限執(zhí)行handleMessage。
            if (msg.what == 1 && !isFinish) {

                //統(tǒng)計(jì)已上傳的大小
                long sum = 0;
                for (long p : mProgresses2.values()) {
                    sum += p;
                }

                //計(jì)算整體的進(jìn)度   注意這里涉及到兩個(gè)long類型相除的問(wèn)題,若不先轉(zhuǎn)換為float類型,則商為0   去尾操作保證了99.9%屬于并沒(méi)有上傳完成的范疇
                int p = 0;
                if (totalLength != 0) {
                    p = (int) Math.floor((float) sum / (float) totalLength * 100);
                }
                //通知View層更新ProgressView的狀態(tài)(自定義的一個(gè)進(jìn)度View)  
                mView.showUploadProgress(p, ProgressView.STATE_LOADING);
                //每0.1秒更新一次UI
                mHandler.sendEmptyMessageDelayed(1, 100);
            } else if (msg.what == 2) {
                //what=2代表上傳成功后  延遲兩秒自動(dòng)回到主頁(yè)的操作
                isFinish = true;
                mView.dismissDialog();
                mView.backToMain();
            } else if (msg.what == 3) {
                //what=3為檢查出圖片有誤,取消上傳的操作
                mView.showToast(msg.obj.toString());
                mView.showUploadProgress(0, ProgressView.STATE_FAILURE);
                isFinish = true;
                //取消訂閱方法  即取消上傳請(qǐng)求
                unSubscribe();
            }
        }
    };

    @Override
    public void unSubscribe() {
        if (mSubscription != null) {
            if (!mSubscription.isUnsubscribed()) {
                mSubscription.unsubscribe();
                isFinish = true;
            }
        } else {
            canStart = false;
            isFinish = true;
        }
    }

上傳成功的即onNext回調(diào)只要執(zhí)行上傳成功的操作:

@Override
                        public void onNext(CheckBaseBean checkBaseBean) {
                            //sign服務(wù)器返回的狀態(tài)值true為成功
                            if (checkBaseBean.sign) {
                                mView.showToast("上傳成功");
                                mView.showUploadProgress(100, ProgressView.STATE_SUCCESS);
                                //延遲兩秒后回調(diào)主頁(yè)
                                mHandler.sendEmptyMessageDelayed(2, 2000);
                                //操作已完成  無(wú)需更新UI  isFinish置為true
                                isFinish = true;
                                //上傳成功后清除本地?cái)?shù)據(jù)庫(kù)緩存
                                clearDB();
                                clearCache();
//                                mView.backToMain();
                            } else {
                                mView.showUploadProgress(0, ProgressView.STATE_FAILURE);
                                mView.showToast("上傳失敗,請(qǐng)反饋給開(kāi)發(fā)人員,謝謝");
                                isFinish = true;
                            }
                        }

小結(jié)

本人不善表達(dá),水平也很臭,大家見(jiàn)諒。由于工作忙,難以擠出時(shí)間擇代碼,直接使用項(xiàng)目中的代碼,因此代碼很臃腫,并不能簡(jiǎn)潔地展示具體過(guò)程。本文僅僅提供一個(gè)思路,具體的實(shí)現(xiàn)相信大家都能自己將其封裝到自己的項(xiàng)目中,畢竟每個(gè)項(xiàng)目的需求不同,實(shí)現(xiàn)方式也會(huì)有差異。

謝謝閱讀,請(qǐng)自行左上返回或右上關(guān)閉。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,936評(píng)論 6 535
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,744評(píng)論 3 421
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人,你說(shuō)我怎么就攤上這事。” “怎么了?”我有些...
    開(kāi)封第一講書人閱讀 176,879評(píng)論 0 381
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我,道長(zhǎng),這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書人閱讀 63,181評(píng)論 1 315
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 71,935評(píng)論 6 410
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書人閱讀 55,325評(píng)論 1 324
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,384評(píng)論 3 443
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書人閱讀 42,534評(píng)論 0 289
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 49,084評(píng)論 1 335
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 40,892評(píng)論 3 356
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 43,067評(píng)論 1 371
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,623評(píng)論 5 362
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,322評(píng)論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書人閱讀 34,735評(píng)論 0 27
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書人閱讀 35,990評(píng)論 1 289
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 51,800評(píng)論 3 395
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 48,084評(píng)論 2 375

推薦閱讀更多精彩內(nèi)容