前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >RxJava +Retrofit 你需要掌握的几个实用技巧

RxJava +Retrofit 你需要掌握的几个实用技巧

作者头像
开发者技术前线
发布2020-11-23 15:27:13
8240
发布2020-11-23 15:27:13
举报

RxJava +Retrofit 你需要掌握的几个技巧,Retrofit中OkHttp缓存,统一对有无网络处理, 异常处理,返回结果问题.

1

取消订阅

一般我们在视图消亡后,无需RxJava再执行,可以直接取消订阅

代码语言:javascript
复制
subscription.unsubscribe() 
//取消发生在IO线程
observable.unsubscribeOn(Schedulers.io());

可用在activity的 onDestroy(), Fragment的 onDestroyView()中调用

还有种场景是借助rxJava请求网络数据,需要网络返回后保存数据并更新UI,这种情况视图已经消亡了必定会导致rxJava出错,导致App闪退,这种我们可以判断前的activity/view是否为空,并是否已showing,如果 两者都不存在,即可无须更新UI。只处理保存数据。

2

订阅问题

需要UI绘制后再进行订阅的场景,防止阻塞UI,我们需要延迟订阅执行。

立即订阅

代码语言:javascript
复制
   observable
         .subscribeOn(Schedulers.io())               
         .observeOn(AndroidSchedulers.mainThread())                       .subscribe(action);

延迟订阅

代码语言:javascript
复制
observable.delay(2, TimeUnit.SECONDS)              
     .subscribeOn(Schedulers.io())               
     .observeOn(AndroidSchedulers.mainThread())              .subscribe(action);

3

基本API

通常我们写接口会有以下定义,增加一个api就必须写一个方法

代码语言:javascript
复制
public interface MyApi {
  @GET("app.php")
  Observable<SouguBean> getSougu(
    @Query("name") String name);
    
  @GET("/getWeather")
  Observable<ResponseBody> getWeather(
     @QueryMap Map<String, String> maps);
}

很多时候每新增一个接口就要写一个api,是不是有很好的方法代替这种情况。

代码语言:javascript
复制
@GET()
<T> Observable<ResponseBody> get(        
    @Url String url,        
    @QueryMap Map<String, T> maps);

我们可以定义一个通用的getApi,将url动态传入,

返回Model定义为ResponseBody, 并将实际参数定义为泛型,不管是更改url,还是服务端返回类型,包括参数个数都可以完美适配,这种方式技术不到位的千万别用,因为Retrofit明确说明接口必须要给定明确类型,悠着点哈!

上层进行通用组装时就可以这样子:

代码语言:javascript
复制
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带来接口适配问题,一用一个准啊…

4

基础Subscriber

很多时候我们需要借用RxJava开启多个observable去读取网络,这是我们对不同Subscriber处理起来比较麻烦,因此统一对Subscriber对网络返回进行处理和, 有无网络做判断,甚至可以根据需求显示加载进度等 构建抽象的BaseSubscribe类,只处理start()onCompleted() ,上层处理时只处理onError()onNext()

代码语言:javascript
复制
 /**
    * 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, "
          当前网络不可用,请检查网络情况", 
          Toast.LENGTH_SHORT).show();              
          // **一定要主动调用下面这一句**
               onCompleted();               
               return;
           }           
           // 显示进度条
           showLoadingProgress();
       }      
       
       @Override
       public void onCompleted() {         
          //关闭等待进度条
          closeLoadingProgress();
       }
    }

这样我们上层调用时只关心成功和失败即可,无需再关心网络情况

代码语言:javascript
复制
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错误统一处理,也可以在BaseSubscriber处理onError(), 然后回调搭到callback上层,具体看自己项目情况而定,可以接着往下看

如果对成功结果进行处理,则可以将ResonseBody加入泛型<Response<T>> , Response一般是包含Code,MSg, Data的,在这里你可以根据判断code来进行业务分发,代码很简单,具体看文章结尾源码即可

如果你觉得目前的返回判断麻烦,也可以定义Response基类

代码语言:javascript
复制
 /**
* 网络返回基类 支持泛型
* 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() 只需统一判断状态码即可

代码语言:javascript
复制
 @Override
 public void onNext(BaseResponse<IpResult> responseBody) {                            
  if (responseBody.isOk()) {                           
  //这里这个ok不是http访问的ok,//是和服务器约定好的成功码 有的人不喜欢可以不用加这个筛选, 也有的人喜欢将业务加到业务回调中,如果不是成功码 也//不走错误回调,也不走成功回调,直走业务回调
代码语言:javascript
复制
     IpResult ip = responseBody.getData();
     Toast.makeText(MainActivity.this, ip.toString(), Toast.LENGTH_LONG).show();
                    
      }
  }

5

错误结果处理问题

通过RXJva的 Func1来进行对原始的Throwable 进行包装转换

我们将原来Throwable 强转成自定义的 ResponeThrowable;

代码语言:javascript
复制
private static class HttpResponseFunc《T》 implements Func1《Throwable, Observable《T》》 {   
  @Override public Observable<T> call(Throwable t) {        
       return Observable.error(ExceptionHandle.handleException(t));
    }
}

ResponeThrowable

代码语言:javascript
复制
public static class ResponeThrowable extends Exception {
    public int code;
    public String message;
   
    public ResponeThrowable(Throwable throwable, int code) {           super(throwable);        
      this.code = code;

    }
}

我们已经处理好强转工作后 继续把 Func1加到Observable 中:

因此这样用observable提供的onErrorResumeNext 则可以将你自定义的Func1 关联到错误处理类中:

代码语言:javascript
复制
  ((Observable) observable).onErrorResumeNext(new HttpResponseFunc<T>());

很可能你感觉有点不理解,这前提你需要了解RxJava的转义符和操 Observable.Transformer 还有Func1

这样我们对服务器返回的错误状态进行了自我的处理,再稍加翻译下便可以达到用户看懂的语言

这个类我参考一叶飘舟同学的案列,我再次做了改进:

ExceptionHandle 错误处理驱动

代码语言:javascript
复制
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;    i
    f (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 = "网络错误";                
                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;   
     /**
     * 网络错误
     */
    public static final int NETWORD_ERROR = 1002;    
    /**
     * 协议出错
     */
    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>中处理异常!

代码语言:javascript
复制
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));
    }
 }
}

最后上层调用就是这样了:

代码语言:javascript
复制
 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 的实现类和子类即可,如果你想要重写BaseSubscriber 的onStat()和onCompleted() 也是可以的, 一般BaseSubscriber只处理公用的处理,或者进行下业务对返回格式检查,具体成功 解析有他的子类(实现类)去做。

注意:如果你不想将业务分发加到错误回调中,也可以这样做: 好比有的人喜欢将业务处理加到业务回调中,如果后台返回的业务码并不成功码的情况下, 不想走错误回调,也不想走成功回调, 想走直走业务回调。

可以这样处理:

在onNext() 中回调一个自定义的抽象的onBusiness(code, masg),将他的子类去实现

代码语言:javascript
复制
 @Override
 public void onNext(BaseResponse<IpResult> responseBody) {                            
     if (!responseBody.isOk()) {                                      //业务分发
       onBusiness(responseBody.getCode, responseBody.getMsg)
                           
      } else {     // 成功回调
           onNext(responseBody.getData())
                            
     }

   }

6

概述缓存问题

公共缓存: 有时候需要在无网络时增加缓存功能,因此给Retrofit加入基础拦截器,来处理缓存问题

代码语言:javascript
复制
/**
    * 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, "当前无网络!", 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加入拦截器

代码语言:javascript
复制
       okHttpClient = new OkHttpClient.Builder()
             .addInterceptor(new BaseInterceptor(headers))
            .addInterceptor(new   CaheInterceptor(context))
            .addNetworkInterceptor(new CaheInterceptor(context))
            .build();

Retrofit 加入okhttpClient

代码语言:javascript
复制
retrofit = new Retrofit.Builder()
                .client(okHttpClient)
                .baseUrl(url)
                .build();

单独缓存:

如果你不想加入公共缓存,想单独对某个api进行缓存,可用Headers来实现,那么可以这样:

代码语言:javascript
复制
;

值得注意的是 下面的两句话也必须加入:

代码语言:javascript
复制
  .addInterceptor(new   CacheInterceptor(context))
  .addNetworkInterceptor(new CacheInterceptor(context))

缓存路径和默认大小

如果想更改okhttp的缓存路劲,可以设置cache的path路径 ,姿势如下

代码语言:javascript
复制
 Cache cache = new Cache(httpCacheDirectory, 10 * 1024 * 1024);

第一个参数是路径,第二个最大缓存大小

代码语言:javascript
复制
  okHttpClient = new OkHttpClient.Builder()
                        .cache(cache)
                        .build();

这样就加入自定义的Cache策略

自定义缓存

如果你不想用okhttp自带的缓存策略,因为这需要服务端配合处理缓存请求头,不然会抛出: HTTP 504 Unsatisfiable Request (only-if-cached)

除了以上修改 Request.cacheControl的方式实现缓存,也可以自定义一个Cahe策略用来实现本地硬缓存。

构建CacheManager,用Url对应Json实现,此类非常简单,你可以自己实现,时间策略可自我加入扩展 在BaseSubscriber进行网络判断,加载缓存数据返回妥妥的;

代码语言:javascript
复制
 @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, "无网络", Toast.LENGTH_SHORT).show();        
        if (isNeedCahe) {
            Toast.makeText(context, "无网络,已智能读取缓存!", Toast.LENGTH_SHORT).show();
            IpResult ipResult = new Gson().fromJson(CaheManager.getjson(url), IpResult.class);
            onNext((T) ipResult);
        }
        onCompleted();
    }

}

7

总结

通过这次的整理,再进行RxJava和Retrofit中 ,所有坑直接添就行,接着上次的介绍,更多介绍请见 Novate。

作 者: Tamic

编 辑: Tamic

来 源:开发者技术前线 链 接:http://www.jianshu.com/p/d7734390895e

精彩推荐

Rxjava与Retrofit相恋,如何优雅的取消请求!

是时候客观评价Retrofit了,这几点你必须明白

Android基于Retrofit2.0 封装的超好用的RetrofitClient工具类

你不知道的Retrofit缓存库RxCache

技术 - 思维 - 感悟

END

本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2017-07-25,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 开发者技术前线 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档