Retrofit & RxJava 實戰(zhàn)篇

一、網(wǎng)絡請求實現(xiàn):輪詢、嵌套坯癣、合并數(shù)據(jù)哼勇、緩存都伪、出錯重試

public class RetrofitRxJavaStudy {

    private static NetworkInterface networkInterface;

    //獲取接口類NetworkInterface并初始化Retrofit
    private static NetworkInterface getRequestInterface() {
        if (networkInterface == null) {
            synchronized (NetworkInterface.class) {
                networkInterface = initRetrofit().create(NetworkInterface.class);
            }
        }
        return networkInterface;
    }

    private static Retrofit initRetrofit() {
        OkHttpClient client = new OkHttpClient.Builder()
                .build();
        Retrofit retrofit = new Retrofit.Builder()
                .baseUrl(NetworkInterface.HOST)//設置網(wǎng)絡請求的Url地址
                .addConverterFactory(GsonConverterFactory.create())//設置數(shù)據(jù)解析器
                .addCallAdapterFactory(RxJava2CallAdapterFactory.create())//支持RxJava平臺
                .client(client)
                .build();
        return retrofit;
    }

    //初始化請求數(shù)據(jù)參數(shù)
    private RequestBody initData() {
        Gson gson = new Gson();
        Map<String, String> map = new HashMap<String, String>();
        map.put("phone", "phone");
        map.put("code", "code");
        String strJson = gson.toJson(map);
        RequestBody body = RequestBody.create(okhttp3.MediaType.parse("application/json; charset=utf-8"), strJson);
        return body;
    }


    //無條件網(wǎng)絡請求輪詢
    private void interval() {
        //initialDelay:第一次延遲時間、period:間隔時間數(shù)字积担、unit:時間單位
        Observable.interval(2, 10, TimeUnit.SECONDS).doOnNext(new Consumer<Long>() {
            @Override
            public void accept(Long aLong) throws Exception {
                //網(wǎng)絡請求具體實現(xiàn)
                getRequestInterface()
                        .registerLogin("login/login", initData())
                        .observeOn(AndroidSchedulers.mainThread())//回調在主線程
                        .subscribeOn(Schedulers.io())
                        .subscribe();//執(zhí)行在io線程
            }
        }).subscribe();

        //start:事件序列起始點陨晶、count:事件數(shù)量、initialDelay:第1次事件延遲發(fā)送時間帝璧、period:間隔時間數(shù)字先誉、unit:時間單位
        Observable.intervalRange(2, 10, 5, 10, TimeUnit.SECONDS).doOnNext(new Consumer<Long>() {
            @Override
            public void accept(Long aLong) throws Exception {
                //網(wǎng)絡請求具體實現(xiàn)code...
            }
        }).subscribe();
    }

    //有條件網(wǎng)絡請求輪詢
    private void repeatWhen() {
        final int[] i = {0};
        getRequestInterface()
                .registerLogin("login/login", initData())
                .repeatWhen(new Function<Observable<Object>, ObservableSource<?>>() {
                    @Override
                    public ObservableSource<?> apply(Observable<Object> objectObservable) throws Throwable {
                        if (i[0] > 3) {
                            // 此處選擇發(fā)送onError事件以結束輪詢
                            return Observable.error(new Throwable("輪詢結束"));
                        }
                        //delay操作符延遲一段時間發(fā)送以實現(xiàn)輪詢間間隔設置
                        return Observable.just(1).delay(5000, TimeUnit.MILLISECONDS);
                    }
                })
                .observeOn(AndroidSchedulers.mainThread())//回調在主線程
                .subscribeOn(Schedulers.io())//執(zhí)行在io線程
                .subscribe(new Consumer<RegisterLogin>() {
                    @Override
                    public void accept(RegisterLogin registerLogin) throws Throwable {
                        i[0]++;
                    }
                });
    }

    //網(wǎng)絡請求嵌套回調
    private void doOnNext() {
        getRequestInterface()
                .registerLogin("login/login", initData())
                .observeOn(AndroidSchedulers.mainThread())//回調在主線程
                .subscribeOn(Schedulers.io())//執(zhí)行在io線程
                .doOnNext(new Consumer<RegisterLogin>() {
                    @Override
                    public void accept(RegisterLogin registerLogin) throws Throwable {
                        //第1次網(wǎng)絡請求成功
                    }
                })
                .observeOn(Schedulers.io())//新觀察者切換到IO線程去發(fā)起請求
                .flatMap(new Function<RegisterLogin, ObservableSource<RegisterLogin>>() {
                    @Override
                    public ObservableSource<RegisterLogin> apply(RegisterLogin registerLogin) throws Throwable {
                        //registerLogin是網(wǎng)絡請求1返回數(shù)據(jù),發(fā)送網(wǎng)絡請求2
                        return getRequestInterface()
                                .registerLogin("login/login", initData());
                    }
                })
                .observeOn(AndroidSchedulers.mainThread())//切換到主線程處理網(wǎng)絡請求2的結果
                .subscribe(new Consumer<RegisterLogin>() {
                    @Override
                    public void accept(RegisterLogin o) throws Throwable {
                        //o是網(wǎng)絡請求2返回數(shù)據(jù)
                    }
                });
    }

    //采用 Merge()操作符合并數(shù)據(jù)
    private void merge() {
        Observable<String> network = Observable.just("網(wǎng)絡");
        Observable<String> file = Observable.just("本地文件");
        Observable.merge(network, file).subscribe(new Observer<String>() {
            @Override
            public void onSubscribe(Disposable d) {
            }

            @Override
            public void onNext(String value) {
            }

            @Override
            public void onError(Throwable e) {
            }

            @Override
            public void onComplete() {
            }
        });
    }

    //采用Zip()操作符合并數(shù)據(jù)(網(wǎng)絡數(shù)據(jù) + 網(wǎng)絡數(shù)據(jù))
    private void zip() {
        final Observable<RegisterLogin> observable1 = getRequestInterface()
                .registerLogin("login/login", initData())
                .subscribeOn(Schedulers.io());//執(zhí)行在io線程
        final Observable<RegisterLogin> observable2 = getRequestInterface()
                .registerLogin("login/login", initData())
                .subscribeOn(Schedulers.io());//執(zhí)行在io線程
        Observable.zip(observable1, observable2, new BiFunction<RegisterLogin, RegisterLogin, Object>() {
            @Override
            public Object apply(RegisterLogin registerLogin, RegisterLogin registerLogin2) throws Throwable {
                return registerLogin.getMessage() + " & " + registerLogin2.getMessage();
            }
        }).observeOn(AndroidSchedulers.mainThread())
                .subscribe(new Consumer<Object>() {
                    @Override
                    public void accept(Object o) throws Throwable {

                    }
                });
    }

    //從磁盤的烁、內存緩存中獲取緩存數(shù)據(jù)firstElement和concat
    private void firstElement() {
        final String memoryCache = null;
        final String diskCache = "從磁盤緩存中獲取數(shù)據(jù)";
        Observable<String> memory = Observable.create(new ObservableOnSubscribe<String>() {
            @Override
            public void subscribe(ObservableEmitter<String> emitter) throws Exception {
                // 先判斷內存緩存有無數(shù)據(jù)
                if (memoryCache != null) { // 若有該數(shù)據(jù)褐耳,則發(fā)送
                    emitter.onNext(memoryCache);
                } else { // 若無該數(shù)據(jù),則直接發(fā)送結束事件
                    emitter.onComplete();
                }
            }
        });

        Observable<String> disk = Observable.create(new ObservableOnSubscribe<String>() {
            @Override
            public void subscribe(ObservableEmitter<String> emitter) throws Exception {
                // 先判斷內存緩存有無數(shù)據(jù)
                if (diskCache != null) { // 若有該數(shù)據(jù)渴庆,則發(fā)送
                    emitter.onNext(diskCache);
                } else { // 若無該數(shù)據(jù)铃芦,則直接發(fā)送結束事件
                    emitter.onComplete();
                }
            }
        });

        Observable<String> observable = getRequestInterface()
                .getBody("login/login");//執(zhí)行在io線程

        Observable.concat(memory, disk, observable)
                //通過firstElement()雅镊,從串聯(lián)隊列中取出并發(fā)送第1個有效事件(Next事件)
                //即依次判斷檢查memory、disk刃滓、network
                .firstElement()
                .subscribe(new Consumer<String>() {
                    @Override
                    public void accept(String s) throws Exception {
                        //最終獲取的數(shù)據(jù)來源
                    }
                });
    }

    // 可重試次數(shù)
    private int maxConnectCount = 10;
    // 當前已重試次數(shù)
    private int currentRetryCount = 0;
    // 重試等待時間
    private int waitRetryTime = 0;

    //網(wǎng)絡請求出錯重試
    private void flatMap() {
        getRequestInterface()
                .registerLogin("login/login", initData())
                .retryWhen(new Function<Observable<Throwable>, ObservableSource<?>>() {
                    @Override
                    public ObservableSource<?> apply(@NonNull Observable<Throwable> throwableObservable) throws Exception {
                        return throwableObservable.flatMap(new Function<Throwable, ObservableSource<?>>() {
                            @Override
                            public ObservableSource<?> apply(@NonNull Throwable throwable) throws Exception {
                                //根據(jù)異常類型選擇是否重試仁烹,屬于IO異常,需重試
                                if (throwable instanceof IOException) {
                                    //當已重試次數(shù) < 設置的重試次數(shù)才重試
                                    if (currentRetryCount < maxConnectCount) {
                                        // 記錄重試次數(shù)
                                        currentRetryCount++;
                                        // 設置等待時間
                                        waitRetryTime = 1000 + currentRetryCount * 1000;
                                        return Observable.just(1).delay(waitRetryTime, TimeUnit.MILLISECONDS);
                                    } else {
                                        // 若重試次數(shù)已 > 設置重試次數(shù)注盈,則不重試
                                        return Observable.error(new Throwable("重試次數(shù)已超過設置次數(shù) = " + currentRetryCount + "晃危,即 不再重試"));
                                    }
                                } else {
                                    // 若發(fā)生的異常不屬于I/O異常,則不重試
                                    return Observable.error(new Throwable("發(fā)生了非網(wǎng)絡異常(非I/O異常)"));
                                }
                            }
                        });
                    }
                })
                .subscribeOn(Schedulers.io())//執(zhí)行在io線程
                .subscribe();
    }
}
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
  • 序言:七十年代末老客,一起剝皮案震驚了整個濱河市僚饭,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌胧砰,老刑警劉巖鳍鸵,帶你破解...
    沈念sama閱讀 211,817評論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異尉间,居然都是意外死亡偿乖,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,329評論 3 385
  • 文/潘曉璐 我一進店門哲嘲,熙熙樓的掌柜王于貴愁眉苦臉地迎上來贪薪,“玉大人,你說我怎么就攤上這事眠副』校” “怎么了?”我有些...
    開封第一講書人閱讀 157,354評論 0 348
  • 文/不壞的土叔 我叫張陵囱怕,是天一觀的道長霍弹。 經(jīng)常有香客問我,道長娃弓,這世上最難降的妖魔是什么典格? 我笑而不...
    開封第一講書人閱讀 56,498評論 1 284
  • 正文 為了忘掉前任,我火速辦了婚禮台丛,結果婚禮上耍缴,老公的妹妹穿的比我還像新娘。我一直安慰自己挽霉,他們只是感情好防嗡,可當我...
    茶點故事閱讀 65,600評論 6 386
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著炼吴,像睡著了一般本鸣。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上硅蹦,一...
    開封第一講書人閱讀 49,829評論 1 290
  • 那天荣德,我揣著相機與錄音闷煤,去河邊找鬼。 笑死涮瞻,一個胖子當著我的面吹牛鲤拿,可吹牛的內容都是我干的。 我是一名探鬼主播署咽,決...
    沈念sama閱讀 38,979評論 3 408
  • 文/蒼蘭香墨 我猛地睜開眼近顷,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了宁否?” 一聲冷哼從身側響起窒升,我...
    開封第一講書人閱讀 37,722評論 0 266
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎慕匠,沒想到半個月后饱须,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,189評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡台谊,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 36,519評論 2 327
  • 正文 我和宋清朗相戀三年蓉媳,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片锅铅。...
    茶點故事閱讀 38,654評論 1 340
  • 序言:一個原本活蹦亂跳的男人離奇死亡酪呻,死狀恐怖,靈堂內的尸體忽然破棺而出盐须,到底是詐尸還是另有隱情玩荠,我是刑警寧澤,帶...
    沈念sama閱讀 34,329評論 4 330
  • 正文 年R本政府宣布丰歌,位于F島的核電站姨蟋,受9級特大地震影響屉凯,放射性物質發(fā)生泄漏立帖。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 39,940評論 3 313
  • 文/蒙蒙 一悠砚、第九天 我趴在偏房一處隱蔽的房頂上張望晓勇。 院中可真熱鬧,春花似錦灌旧、人聲如沸绑咱。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,762評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽描融。三九已至,卻和暖如春衡蚂,著一層夾襖步出監(jiān)牢的瞬間窿克,已是汗流浹背骏庸。 一陣腳步聲響...
    開封第一講書人閱讀 31,993評論 1 266
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留年叮,地道東北人具被。 一個月前我還...
    沈念sama閱讀 46,382評論 2 360
  • 正文 我出身青樓,卻偏偏與公主長得像只损,于是被迫代替她去往敵國和親一姿。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 43,543評論 2 349

推薦閱讀更多精彩內容