Rxjava +Retrofit 你需要掌握的幾個技巧,Retrofit緩存锅减,統(tǒng)一對有無網(wǎng)絡(luò)處理, 異常處理糖儡,返回結(jié)果問題

本文出處 :Tamic
文/ http://www.reibang.com/p/b1979c25634f

Rxjava +Rterofit 需要掌握的幾個技巧

這里寫圖片描述

RxJava入門和詳解請移步 比較有名的《RxJAVA詳解》,這里繼續(xù)前篇一系列的介紹一些容易忽略的技巧.

Retrofit+RxJava結(jié)合系列請閱讀:


取消訂閱

一般我們在視圖消亡后拴疤,無需RxJava再執(zhí)行,可以直接取消訂閱

subscription.unsubscribe()

 observable.unsubscribeOn(Schedulers.io());

可用在activity的 onDestroy(), Fragment的 onDestroyView()中調(diào)用

還有種場景是借助rxJava請求網(wǎng)絡(luò)數(shù)據(jù)独泞,需要網(wǎng)絡(luò)返回后保存數(shù)據(jù)并更新UI呐矾,這種情況視圖已經(jīng)消亡了必定會導(dǎo)致rxJava出錯,導(dǎo)致App閃退懦砂,這種我們可以判斷前的activity/view是否為空蜒犯,并是否已showing,如果
兩者都不存在荞膘,即可無須更新UI罚随。只處理保存數(shù)據(jù)即可。

訂閱問題

需要UI繪制后再進行訂閱的場景羽资,防止阻塞UI淘菩,我們需要延遲訂閱執(zhí)行。

立即訂閱;

         observable
               .subscribeOn(Schedulers.io())
               .observeOn(AndroidSchedulers.mainThread())
               .subscribe(action);

延遲訂閱

observable.delay(2, TimeUnit.SECONDS)
               .subscribeOn(Schedulers.io())
               .observeOn(AndroidSchedulers.mainThread())
               .subscribe(action);

基礎(chǔ)ApiService

通常我們寫接口會有以下定義屠升,增加一個api就必須寫一個方法

public interface MyApi {

@GET("app.php")
Observable<SouguBean> getSougu(@Query("name") String name);

@GET("/getWeather")
Observable<ResponseBody> getWeather(@QueryMap Map<String, String> maps);
}

很多時候每新增一個接口就要寫一個api潮改,是不是有很好的方法代替這種情況。

@GET()
<T> Observable<ResponseBody> get(
        @Url String url,
        @QueryMap Map<String, T> maps);

我們可以定義一個通用的getApi腹暖,將url動態(tài)傳入汇在,返回Modle定義為ResponseBody, 并將實際參數(shù)定義為泛型脏答,不管是更改url糕殉,還是服務(wù)端返回類型,包括參數(shù)個數(shù)都可以完美適配以蕴,這種方式技術(shù)不到位的千萬別用糙麦,因為Retrofit明確說明接口必須要給定明確類型,悠著點哈丛肮!

上層進行通用組裝時就可以這樣子:

   public <T> T get(String url, Map<String, T> maps, BaseSubscriber<ResponseBody> subscriber) {
return (T) apiManager.get(url, maps)
.compose(schedulersTransformer)
.compose(handleErrTransformer())
.subscribe(subscriber);
}

看不懂?看不懂不算奇怪魄缚,源碼可以去文章末尾下載研究宝与,這里只是列舉了一下焚廊。這種方式很適合從HttpClent遷移到Retrofit帶來接口適配問題,一用一個準(zhǔn)啊…

基礎(chǔ)Subscriber

很多時候我們需要借用RxJava開啟多個observable去讀取網(wǎng)絡(luò)习劫,這是我們對不同Subscriber處理起來比較麻煩咆瘟,因此統(tǒng)一對Subscriber對網(wǎng)絡(luò)返回進行處理和, 有無網(wǎng)絡(luò)做判斷诽里,甚至可以根據(jù)需求顯示加載進度等
構(gòu)建抽象的BaseSubscribe類袒餐,只處理start()onCompleted() ,上層處理時只處理onError()onNext()

 /**
    * BaseSubscriber
    * Created by Tamic on 2016-7-15.
    */
   public abstract class BaseSubscriber<T> extends Subscriber<T> {

       private BaseActivity context;

       public BaseSubscriber(BaseActivity context) {
           this.context = context;
       }

       @Override
       public void onStart() {
           super.onStart();

           if (!NetworkUtil.isNetworkAvailable(context)) {

               Toast.makeText(context, "當(dāng)前網(wǎng)絡(luò)不可用谤狡,請檢查網(wǎng)絡(luò)情況", Toast.LENGTH_SHORT).show();
              // **一定要主動調(diào)用下面這一句**
               onCompleted();

               return;
           }
           // 顯示進度條
           showLoadingProgress();
       }

       @Override
       public void onCompleted() {
          //關(guān)閉等待進度條
          closeLoadingProgress();

       }


    }

這樣我們上層調(diào)用時只關(guān)心成功和失敗即可灸眼,無需再關(guān)心網(wǎng)絡(luò)情況

```
observable..subscribeOn(Schedulers.io())
               .observeOn(AndroidSchedulers.mainThread())
               .subscribe(new BaseSubscriber<ResponseBody>(MainActivity.this) {

                @Override
                public void onError(Throwable e) {
                   Toast.makeText(MainActivity.this, e.getMessage(), Toast.LENGTH_LONG).show();
                }

                @Override
                public void onNext(ResponseBody responseBody) {
                    Toast.makeText(MainActivity.this, responseBody.toString(), Toast.LENGTH_LONG).show();
                }
            });
  );

如果想對Error錯誤統(tǒng)一處理,也可以在BaseSubscriber處理onError(), 然后回調(diào)搭到callback上層墓懂,具體看自己項目情況而定焰宣,可以接著往下看

如果對成功結(jié)果進行處理,則可以將ResonseBody加入泛型<Response<T>> , Response一般是包含Code捕仔,MSg匕积, Data的,在這里你可以根據(jù)判斷code來進行業(yè)務(wù)分發(fā)榜跌,代碼很簡單闪唆,具體看文章結(jié)尾源碼即可

如果你覺得目前的返回判斷麻煩,也可以定義Response基類

 /**
* 網(wǎng)絡(luò)返回基類 支持泛型
* Created by Tamic on 2016-06-06.
*/

public class BaseResponse<T> {

  private int code;
  private String msg;
  private T data;

 public int getCode() {
     return code;
 }

public void setCode(int code) {
    this.code = code;
}

public String getMsg() {
    return msg;
}

public void setMsg(String msg) {
    this.msg = msg;
}

public T getData() {
    return data;
}

public void setData(T data) {
    this.data = data;
}

public boolean isOk() {
    return code == 0;
}

}

這樣我們在onNext() 只需統(tǒng)一判斷狀態(tài)碼即可

                       @Override
                        public void onNext(BaseResponse<IpResult> responseBody) {

                            if (responseBody.isOk()) {
                           //這里這個ok不是http訪問的ok钓葫,

//是和服務(wù)器約定好的成功碼 有的人不喜歡可以不用加這個篩選悄蕾, 也有的人喜歡將業(yè)務(wù)加到業(yè)務(wù)回調(diào)中,如果不是成功碼 也//不走錯誤回調(diào)瓤逼,也不走成功回調(diào)笼吟,直走業(yè)務(wù)回調(diào)

                                IpResult ip = responseBody.getData();
                                Toast.makeText(MainActivity.this, ip.toString(), Toast.LENGTH_LONG).show();
                            }

                        }

錯誤結(jié)果問題

通過RXJva的 Func1來進行對原始的Throwable 進行包裝轉(zhuǎn)換

我們將原來Throwable 強轉(zhuǎn)成自定義的 ResponeThrowable;

private static class HttpResponseFunc《T》 implements Func1《Throwable, Observable《T》》 {
    @Override public Observable<T> call(Throwable t) {
        return Observable.error(ExceptionHandle.handleException(t));
    }
}

ResponeThrowable

public static class ResponeThrowable extends Exception {
    public int code;
    public String message;

    public ResponeThrowable(Throwable throwable, int code) {
        super(throwable);
        this.code = code;

    }
}

我們已經(jīng)處理好強轉(zhuǎn)工作后 繼續(xù)把 Func1加到Observable 中:

因此這樣用observable提供的onErrorResumeNext 則可以將你自定義的Func1 關(guān)聯(lián)到錯誤處理類中:

  ((Observable) observable).onErrorResumeNext(new HttpResponseFunc<T>());

很可能你感覺有點不理解霸旗,這前提你需要了解RxJava的轉(zhuǎn)義符和操 Observable.Transformer
還有Func1

這樣我們對服務(wù)器返回的錯誤狀態(tài)進行了自我的處理贷帮,再稍加翻譯下便可以達到用戶看懂的語言

這個類我參考一葉扁舟同學(xué)的案列,我再次做了改進:

ExceptionHandle 錯誤處理驅(qū)動

public class ExceptionHandle {

 private static final int UNAUTHORIZED = 401;
 private static final int FORBIDDEN = 403;
 private static final int NOT_FOUND = 404;
 private static final int REQUEST_TIMEOUT = 408;
 private static final int INTERNAL_SERVER_ERROR = 500;
 private static final int BAD_GATEWAY = 502;
 private static final int SERVICE_UNAVAILABLE = 503;
 private static final int GATEWAY_TIMEOUT = 504;

 public static ResponeThrowable handleException(Throwable e) {
    ResponeThrowable ex;
    if (e instanceof HttpException) {
        HttpException httpException = (HttpException) e;
        ex = new ResponeThrowable(e, ERROR.HTTP_ERROR);
        switch (httpException.code()) {
            case UNAUTHORIZED:
            case FORBIDDEN:
            case NOT_FOUND:
            case REQUEST_TIMEOUT:
            case GATEWAY_TIMEOUT:
            case INTERNAL_SERVER_ERROR:
            case BAD_GATEWAY:
            case SERVICE_UNAVAILABLE:
            default:
                ex.message = "網(wǎng)絡(luò)錯誤";
                break;
        }
        return ex;
    } else if (e instanceof ServerException) {
        ServerException resultException = (ServerException) e;
        ex = new ResponeThrowable(resultException, resultException.code);
        ex.message = resultException.message;
        return ex;
    } else if (e instanceof JsonParseException
            || e instanceof JSONException
            || e instanceof ParseException) {
        ex = new ResponeThrowable(e, ERROR.PARSE_ERROR);
        ex.message = "解析錯誤";
        return ex;
    } else if (e instanceof ConnectException) {
        ex = new ResponeThrowable(e, ERROR.NETWORD_ERROR);
        ex.message = "連接失敗";
        return ex;
    } else if (e instanceof javax.net.ssl.SSLHandshakeException) {
        ex = new ResponeThrowable(e, ERROR.SSL_ERROR);
        ex.message = "證書驗證失敗";
        return ex;
    }
    else {
        ex = new ResponeThrowable(e, ERROR.UNKNOWN);
        ex.message = "未知錯誤";
        return ex;
    }
}

/**
 * 約定異常
 */
class ERROR {
    /**
     * 未知錯誤
     */
    public static final int UNKNOWN = 1000;
    /**
     * 解析錯誤
     */
    public static final int PARSE_ERROR = 1001;
    /**
     * 網(wǎng)絡(luò)錯誤
     */
    public static final int NETWORD_ERROR = 1002;
    /**
     * 協(xié)議出錯
     */
    public static final int HTTP_ERROR = 1003;

    /**
     * 證書出錯
     */
    public static final int SSL_ERROR = 1005;
}

public static class ResponeThrowable extends Exception {
    public int code;
    public String message;

    public ResponeThrowable(Throwable throwable, int code) {
        super(throwable);
        this.code = code;

    }
 }

 public class ServerException extends RuntimeException {
    public int code;
    public String message;
 }
}

接著可以在 BaseSubscriber<T>中處理異常拉

public abstract class BaseSubscriber<T> extends Subscriber<T> {

private Context context;


public BaseSubscriber(Context context) {
    this.context = context;
}

@Override
public void onError(Throwable e) {
    Log.e("Tamic", e.getMessage());
    // todo error somthing

    if(e instanceof ExceptionHandle.ResponeThrowable){
        onError((ExceptionHandle.ResponeThrowable)e);
    } else {
        onError(new ExceptionHandle.ResponeThrowable(e, ExceptionHandle.ERROR.UNKNOWN));
    }
 }
}

最后上層調(diào)用就是這樣了:

 RetrofitClient.getInstance(MainActivity.this).createBaseApi().getData(new   BaseSubscriber<IpResult>(MainActivity.this) {

                @Override
                public void onError(ResponeThrowable e) {
                   // 處理翻譯后異常诱告。
                    Log.e("Tamic", e.code + " "+ e.message);
                    Toast.makeText(MainActivity.this, e.message, Toast.LENGTH_LONG).show();

                }

                @Override
                public void onNext(IpResult responseBody) {
                    Toast.makeText(MainActivity.this, responseBody.toString(), Toast.LENGTH_LONG).show();
                }
            }, "21.22.11.33");

值的注意的是上層使用BaseSubscriber 的實現(xiàn)類和子類即可撵枢,如果你想要重寫B(tài)aseSubscriber 的onStat()和onCompleted() 也是可以的, 一般BaseSubscriber只處理公用的處理精居,或者進行下業(yè)務(wù)對返回格式檢查锄禽,具體成功
解析有他的子類(實現(xiàn)類)去做。

注意:如果你不想將業(yè)務(wù)分發(fā)加到錯誤回調(diào)中靴姿,也可以這樣做: 好比有的人喜歡將業(yè)務(wù)處理加到業(yè)務(wù)回調(diào)中沃但,如果后臺返回的業(yè)務(wù)碼并不成功碼的情況下, 不想走錯誤回調(diào)佛吓,也不想走成功回調(diào)宵晚, 想走直走業(yè)務(wù)回調(diào)垂攘。

可以這樣處理:

在onNext() 中回調(diào)一個自定義的抽象的onBusiness(code, masg),將他的子類去實現(xiàn)

                       @Override
                        public void onNext(BaseResponse<IpResult> responseBody) {

                            if (!responseBody.isOk()) {
                                  //業(yè)務(wù)分發(fā)
                                 onBusiness(responseBody.getCode, responseBody.getMsg)
                            } else {
                                  // 成功回調(diào)
                                  onNext(responseBody.getData())
                            }

                        }

緩存問題

公共緩存
有時候需要在無網(wǎng)絡(luò)時增加緩存功能淤刃,因此給Retrofit加入基礎(chǔ)攔截器晒他,來處理緩存問題

/**
    * BaseInterceptor
    * Created by Tamic on 2016-7-15.
    */
public class BaseInterceptor implements Interceptor{
    private Map<String, String> headers;
    private Context context;
    public BaseInterceptor(Map<String, String> headers, Context context) {
        this.headers = headers;
        this.context = context;
    }

    @Override
    public Response intercept(Chain chain) throws IOException {

        Request.Builder builder = chain.request()
                .newBuilder();
        builder.cacheControl(CacheControl.FORCE_CACHE).url(chain.request().url())
        .build();

        if (!NetworkUtil.isNetworkAvailable(context)) {

            ((Activity)context).runOnUiThread(new Runnable() {
                @Override
                public void run() {
                    Toast.makeText(context, "當(dāng)前無網(wǎng)絡(luò)!", Toast.LENGTH_SHORT).show();
                }
            });
        }
        if (headers != null && headers.size() > 0) {
            Set<String> keys = headers.keySet();
            for (String headerKey : keys) {
                builder.addHeader(headerKey, headers.get(headerKey)).build();
            }
        }

        if (NetworkUtil.isNetworkAvailable(context)) {
            int maxAge = 60; // read from cache for 60 s
            builder
                    .removeHeader("Pragma")
                    .addHeader("Cache-Control", "public, max-age=" + maxAge)
                    .build();
        } else {
            int maxStale = 60 * 60 * 24 * 14; // tolerate 2-weeks stale
            builder
                    .removeHeader("Pragma")
                    .addHeader("Cache-Control", "public, only-if-cached, max-stale=" + maxStale)
                    .build();
        }
        return chain.proceed(builder.build());

    }
}

okHttpClient加入攔截器

       okHttpClient = new OkHttpClient.Builder()
             .addInterceptor(new BaseInterceptor(headers))
            .addInterceptor(new   CaheInterceptor(context))
            .addNetworkInterceptor(new CaheInterceptor(context))
            .build();

Retrofit 加入okhttpClient

retrofit = new Retrofit.Builder()
                .client(okHttpClient)
                .baseUrl(url)
                .build();

單獨緩存

如果你不想加入公共緩存,想單獨對某個api進行緩存逸贾,可用Headers來實現(xiàn)陨仅,那么可以這樣:

@Headers("Cache-Control : public, max-age = 3600")
@GET("service/getIpInfo.php")
Observable<BaseResponse<IpResult>> getData(@Query("ip") String ip);

值得注意的是 下面的兩句話也必須加入:

             .addInterceptor(new   CaheInterceptor(context))
             .addNetworkInterceptor(new CaheInterceptor(context))

緩存路徑和默認(rèn)大小

如果想更改okhttp的緩存路勁,可以設(shè)置cache的path路徑 铝侵,姿勢如下

 Cache cache = new Cache(httpCacheDirectory, 10 * 1024 * 1024);

第一個參數(shù)是路徑灼伤,第二個最大緩存大小

                      okHttpClient = new OkHttpClient.Builder()
                          .cache(cache)
                           .build();

這樣就加入自定義的Cache策略

自定義緩存

如果你不想用okhttp自帶的緩存策略,因為這需要服務(wù)端配合處理緩存請求頭哟沫,不然會拋出: HTTP 504 Unsatisfiable Request (only-if-cached)

除了以上修改 Request.cacheControl的方式實現(xiàn)緩存饺蔑,也可以自定義一個Cahe策略用來實現(xiàn)本地硬緩存。

構(gòu)建CaheManager嗜诀,用Url對應(yīng)Json實現(xiàn)猾警,此類非常簡單,你可以自己實現(xiàn)隆敢,時間策略可自我加入擴展
在BaseSubscriber進行網(wǎng)絡(luò)判斷发皿,加載緩存數(shù)據(jù)返回妥妥的;

 @Override
 public void onStart() {
    super.onStart();

    Toast.makeText(context, "http is start", Toast.LENGTH_SHORT).show();

    // todo some common as show loadding  and check netWork is NetworkAvailable
    // if  NetworkAvailable no !   must to call onCompleted
    if (!NetworkUtil.isNetworkAvailable(context)) {
        Toast.makeText(context, "無網(wǎng)絡(luò)", Toast.LENGTH_SHORT).show();

        if (isNeedCahe) {
            Toast.makeText(context, "無網(wǎng)絡(luò)拂蝎,已智能讀取緩存穴墅!", Toast.LENGTH_SHORT).show();
            IpResult ipResult = new Gson().fromJson(CaheManager.getjson(url), IpResult.class);
            onNext((T) ipResult);
        }
        onCompleted();
    }

}

總結(jié)

通過這次的整理,再進行RxJava和Retrofit中 温自,所有坑直接添就行玄货,接著上次的介紹,筆者進行新框架開發(fā)novate已快接近尾聲悼泌,估計本月就能和大家見面松捉,敬請繼續(xù)關(guān)注!

Retrofit 2.0系列請閱讀

第一時間獲取技術(shù)文章請關(guān)注掃碼微信公眾號!

開發(fā)者技術(shù)前線

源 碼https://github.com/Tamicer/Novate

本文已在版權(quán)印備案营密,如需轉(zhuǎn)載請訪問版權(quán)印械媒。66790509

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市评汰,隨后出現(xiàn)的幾起案子滥沫,更是在濱河造成了極大的恐慌侣集,老刑警劉巖键俱,帶你破解...
    沈念sama閱讀 206,839評論 6 482
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件兰绣,死亡現(xiàn)場離奇詭異,居然都是意外死亡编振,警方通過查閱死者的電腦和手機缀辩,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,543評論 2 382
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來踪央,“玉大人臀玄,你說我怎么就攤上這事〕澹” “怎么了健无?”我有些...
    開封第一講書人閱讀 153,116評論 0 344
  • 文/不壞的土叔 我叫張陵,是天一觀的道長液斜。 經(jīng)常有香客問我累贤,道長,這世上最難降的妖魔是什么少漆? 我笑而不...
    開封第一講書人閱讀 55,371評論 1 279
  • 正文 為了忘掉前任臼膏,我火速辦了婚禮,結(jié)果婚禮上示损,老公的妹妹穿的比我還像新娘渗磅。我一直安慰自己,他們只是感情好检访,可當(dāng)我...
    茶點故事閱讀 64,384評論 5 374
  • 文/花漫 我一把揭開白布始鱼。 她就那樣靜靜地躺著,像睡著了一般脆贵。 火紅的嫁衣襯著肌膚如雪医清。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,111評論 1 285
  • 那天丹禀,我揣著相機與錄音状勤,去河邊找鬼。 笑死双泪,一個胖子當(dāng)著我的面吹牛持搜,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播焙矛,決...
    沈念sama閱讀 38,416評論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼葫盼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了村斟?” 一聲冷哼從身側(cè)響起贫导,我...
    開封第一講書人閱讀 37,053評論 0 259
  • 序言:老撾萬榮一對情侶失蹤抛猫,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后孩灯,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體闺金,經(jīng)...
    沈念sama閱讀 43,558評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,007評論 2 325
  • 正文 我和宋清朗相戀三年峰档,在試婚紗的時候發(fā)現(xiàn)自己被綠了败匹。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 38,117評論 1 334
  • 序言:一個原本活蹦亂跳的男人離奇死亡讥巡,死狀恐怖掀亩,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情欢顷,我是刑警寧澤槽棍,帶...
    沈念sama閱讀 33,756評論 4 324
  • 正文 年R本政府宣布,位于F島的核電站抬驴,受9級特大地震影響炼七,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜怎爵,卻給世界環(huán)境...
    茶點故事閱讀 39,324評論 3 307
  • 文/蒙蒙 一特石、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧鳖链,春花似錦姆蘸、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,315評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至灌侣,卻和暖如春推捐,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背侧啼。 一陣腳步聲響...
    開封第一講書人閱讀 31,539評論 1 262
  • 我被黑心中介騙來泰國打工牛柒, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人痊乾。 一個月前我還...
    沈念sama閱讀 45,578評論 2 355
  • 正文 我出身青樓皮壁,卻偏偏與公主長得像,于是被迫代替她去往敵國和親哪审。 傳聞我的和親對象是個殘疾皇子蛾魄,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 42,877評論 2 345

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