对Retrofit进行简单封装,提供良好适配节点

在写这个框架的时候,技术Leader跟我强调过一句话:’封装是为了更方便使用,但不要为了封装而封装‘
也正是因为这句话,我在写完beta版本时,又重新审视了一遍,果然发现不少可以精进的地方,最终才有了现在这样清晰结构,感受颇深。

本篇是在几个项目都使用Rxjava 、Retrofit+OkHttp后总结的知识与经验的基础上,综合考虑过项目开发需求、开发使用习惯后而产生的封装思路。

先放一张框架结构图

tbretrofitV2.0.png

这是一次常规请求从发起到结束的整个过程,绿框中的内容就是需要封装来做的事情。
第一次画这个,很蹩脚,东西不多还算清晰吧 哈哈

如果你更关心OkHttp 如何使用Interceptor 来解决 session ,token 过期后 同步重新获取再继续原本的请求,可以参照我的实例 SignInvalidInterceptor.java

封装后解决的核心问题:

1.简化接口的冗余代码及线程调度代码
2.合理的释放线程和Context解绑释放
3.为加解密提供合理易拆分的入口
4.提供可选择的多种缓存模式
5.更清晰的拦截网络异常,优雅的下发异常信息

使用示例
GET:

      HttpUtils.getHttpApi() .get(GITHUB_RESTFUL, new HttpCallBack<GithubEntity>(RxHttpTest.this) {
                    @Override
                    public void onSuccess (GithubEntity githubEntity) {
                        printLog("githubEntity:" + githubEntity.getName());

                    }
                    @Override
                    public void onFailure (int errorCode, String message) {
                        printLog("onFailure  errorCod:" + errorCode + "  errorMsg:" + message);
                    }
                    @Override
                    public CacheModel cacheModel () {
                        return CacheModel.NORMAL;
                    }

                });
    }

POST使用:

   HttpUtils.getHttpApi() .postJson(API.loginUrl, PostDataUtils.getSiginParameter(), new HttpCallBack<SiginResponseBean>(RxHttpTest.this) {
                    @Override
                    public void onSuccess (SiginResponseBean s) {
                    }
                    @Override
                    public CacheModel cacheModel () {
                        return CacheModel.NORMAL;
                    }
                });

看起来好像其实没什么区别,几乎和原来的Retrofit 接口差不多,并不影响使用习惯和代码风格

封装一个库不仅仅是为了减少重复代码,也应该解决一些使用中的问题
下面记录此次加入Rxjava到其中后产生的新思路

一、为什么这样封装以及如何用封装解决

1.用过Retrofit +Rxjava 的都知道,需要为不同接口写不同的Service,使用GsonConvertfactory可以在每个接口返回值直接返回ResultObject。 但这样面临的问题是什么呢?

1).我们需要为每一个接口都在Service 层多加一个接口,想我之前直接使用Service。项目比较大单个Service中至少写了60多个方法,它仅仅是一个接口,加上必要注释大概有300多行。这其中包含大量的重复代码,比如一样的返回值,一样的ip引用,仅仅是对应后台的接口名不同,这显然需要处理掉
解决思路:Retrofit 提供的接口非常丰富,但常用的无非就是Get 、Post、Formdata、和支持文件的Multipart类型。
下面是准备的常用接口:

为什么是Response<String>而不是Object下文会有解释

    @GET
    Observable<Response<String>> get(@Header("Cache-Control") String cacheControl, @Url String url);
    @GET
    Observable<Response<String>> get(@Header("Cache-Control") String cacheControl,@Url String url, @QueryMap Map<String, Object> map);
    @POST
    @FormUrlEncoded
    Observable<Response<String>> postForm(@Header("Cache-Control") String cacheControl,@Url String url, @FieldMap Map<String, Object> map);
    @POST
    @Headers("Content-Type:application/json;")
    Observable<Response<String>> postJson(@Header("Cache-Control") String cacheControl,@Url String url, @Body Object body);
    @POST
    Observable<Response<String>> postIndependent(@Header("Cache-Control") String cacheControl,@Url String url, @Body RequestBody body);
    @POST
    Observable<Response<String>> postFormDataFiles(@Header("Cache-Control") String cacheControl,@Url String url, @Body MultipartBody body);

这里为了方便阅读,去掉了注释,如果有对Retrofit支持的接口类型不熟悉的可以自信查阅后再阅读本篇。
为了保证良好的拓展,专门准备了支持RequestBody作为最高级支持的参数类型

2).使用RxJava 是为了方便管理线程和数据转换,为此需要为每一个接口都写一遍线程调度,处理异常和管理内存释放。但其实通常在Android 中接口请求都是异步的,并不是每一个接口都需要去更换线程调用方式和数据转换。
解决思路:既然线程调度可以保持不变,那就用一个模式搞定他,有了问题1中的接口,就需要一个类似于港口的类来负责转接(可以理解为代理,但他并不是代理模式)。
源码实例:

public interface HttpApi {
    void get(String url,HttpResponseListener responseListener);
    void get(String url, String[] values,HttpResponseListener responseListener);
    void get(String url, Map<String, Object> map,HttpResponseListener responseListener);
    void postJson(String url, JsonBody json,HttpResponseListener responseListener);
    void postRequestBody(String url, RequestBody body,HttpResponseListener responseListener);
    void postFormData(String url, Map<String, Object> map ,HttpResponseListener responseListener);
    void postFormDataFiles(String url, Map<String, Object> map, List<File> files, MediaType contentType,HttpResponseListener responseListener);
}


final public class RxHttpApiImpl implements HttpApi {
 代码省略······
     private  void subscribe (final Observable<Response<String>> observable, final HttpResponseListener responseListener) {
        if (null == observable) return;
        final Subscriber<Response<String>> subscriber = new ResponseHandler(responseListener);
        observable.subscribeOn(Schedulers.io())//被观察者创建线程 事件产生的线程 变量X
                .unsubscribeOn(Schedulers.io())
                .observeOn(AndroidSchedulers.mainThread())//观察者接受回调线程 事件接受线程 应变量Y
                .retryWhen(new RetryWhenTimeout())
                .doOnSubscribe(new Action0() {
                    @Override
                    public void call () {
                        httpTaskManagement.addSubscription(responseListener.getContext(), subscriber);
                    }
                })
                .doOnUnsubscribe(new Action0() {
                    @Override
                    public void call () {
                        httpTaskManagement.remove(responseListener.getContext());
                    }
                })
                .subscribe(subscriber);
    }

  @Override
    public void get (String url, HttpResponseListener responseListener) {
        subscribe(apiService.get(checkCacheModel(responseListener.cacheModel()),url), responseListener);
    }

 代码省略······
}

HttpApi 是面向外部的接口,而在实现类RxHttpApiImpl则负责实现统一的线程调度和解绑
这里解绑包含Subscriber解绑和Context与网络框架解绑。
与Context解绑就是上面doOnUnsubscr中用到的HttpTaskmanagement处理的。
代码如下:

    代码省略······
    @Override
    public void unSubscribe(Object tag) {
        if (null == tasks || tasks.isEmpty()) return;
        RxHttpLog.printI("RxHttpTaskmanagement", "执行取消订阅 key:" + tag.hashCode());
        Subscription subscription = tasks.get(tag);
        if (null != subscription && !subscription.isUnsubscribed()) {
            subscription.unsubscribe();
            RxHttpLog.printI("RxHttpTaskmanagement", "取消了订阅 key:" + tag.hashCode());
            tasks.remove(tag);
        }
    }

    代码省略······

Subscriber解绑:

   @Override
    public void onStart () {
        super.onStart();
        responseListener.onStart();
        //拦截无网络可用
        if (!NetworkStatusUtils.networkIsConnected(responseListener.getContext())) {
            responseListener.onFailure(HttpCode.CODE_NO_INTERNET, "网络不可用");
            unsubscribe();
            onCompleted();
        }
    }

3).因为使用了GsonConvertFactory,所以入参和返回都是JSON格式的。而且必须是Object子类去做,那如果要加密怎么办?显然这样是无法满足的,只想让他作为网络请求的桥梁而非限制如何使用
这里有两种思路去插入加密解密的地方:
a.一个是直接为OkHttp添加Intercepter
这种方法比较死板,容易出错。因为拦截器我们拦截到的是Retrofit中编码后的数据,这样并不利于我们直接操作。通常一个项目中,我们使用的client为了保持一致都是单例的,添加了加密的拦截器有加密的无加密的就无法拆分。所以我并没有实践这种思路
b.重写Convertfactory
在传给OkHttp 之前就进行加密操作,在OkHttp正常返回后进行解密。
加解密我并没有世界操作,但我大概肯定下面的方法是可行的

加密处
GsonRequestBodyConverter源码 :

   @Override 
    public RequestBody convert(T value) throws IOException {
        Buffer buffer = new Buffer();
        Writer writer = new OutputStreamWriter(buffer.outputStream(), UTF_8);
        JsonWriter jsonWriter = gson.newJsonWriter(writer);
        adapter.write(jsonWriter, value);
        jsonWriter.close();
        return RequestBody.create(MEDIA_TYPE, buffer.readByteString());
    }

可以看到这里重新进行了编码,那我们只用对buffer进行加密即可

解密处
StringResponseBodyConverter:

 @Override
    public String convert (ResponseBody value) throws IOException {
        //这里不用拦截 contentLength ==-1?: 源码中转string 时已经判断了
        RxHttpLog.printI("StringResponseBodyConverter","ResponseBody:"+value.contentLength());
        try {
            return value.string();
        } finally {
            value.close();
        }
    }

4.)关于缓存的实现思路
a.为Okhttpclient 添加Intercepter
死板 不利于实现有缓 无缓两种方案
b.利用Retroift支持@Header注解在接口出作为参数添加Cache-Control
实现代码:

  @GET
    Observable<Response<String>> get(@Header("Cache-Control") String cacheControl, @Url String url);
public class CacheConfig {

    /**
     * CacheControl.Builder :
     - noCache();//不使用缓存,用网络请求
     - noStore();//不使用缓存,也不存储缓存
     - onlyIfCached();//只使用缓存
     - noTransform();//禁止转码
     - maxAge(10, TimeUnit.MILLISECONDS);//超过 maxAge 将走网络。
     - maxStale(10, TimeUnit.SECONDS);//超过 maxStale 缓存将不可用
       但依然由 maxAge决定是否走网络,所以 精良让maxAge<maxStale 避免返回 空结果

     - minFresh(10, TimeUnit.SECONDS);//超时时间为当前时间加上10秒钟。

     */


    /**
     * 不考虑缓存,直接走网络,但会存储响应到缓存区
     * @return
     */
    public static String forceNetWork () {

        return CacheControl.FORCE_NETWORK.toString();
    }

    /**
     * 不使用缓存,并且不会存储响应到缓存区
     * @return
     */
    public static String forceNetWorkAndNoStore () {
        return new CacheControl.Builder()
                .noCache()
                .noStore()
                .build().toString();
    }

    /**
     * 直接读取缓存区
     * 如果已有缓存,则将缓存可用时间设置为 Integer.MAX_VALUE
     * 如果缓存区无数据则返回null  code: 504-unsatisfiable request
     * @return
     */
    public static String forceCache () {

        return CacheControl.FORCE_CACHE.toString();
    }


    /**
     * 完全交给Cache-Control:value
     * 由value来决定 重新请求还是使用缓存
     * 另外:Cache-Control 区分 client  与 server,
     * 也就是服务器对应的 expires 有效时间   client max-age
     * 并且由 client 为准 server 为辅
     *
     * max-age client自主决定缓存有效期 超过有效期才请求网络
     * 但是如果服务器依然返回304 则将继续使用缓存
     *
     * @return
     */
    public static String normal () {
        return new CacheControl.Builder()
                .maxAge(30, TimeUnit.SECONDS)
                .build().toString();
    }

    /**
     * 针对近似永久缓存数据,使用如下策略
     * 接口配置 只使用缓存,加入无缓存 504 拦截器
     * 为此次任务重新配置cache-control 在首次读取时从网络下载一次
     * @return
     */
    public static String forever () {
        return new CacheControl.Builder()
                .maxAge(1, TimeUnit.SECONDS)
                .maxStale(Integer.MAX_VALUE,TimeUnit.SECONDS)
                .build().toString();
    }
}

5).关于OkhttpClient网络异常(不包含40x 50x 这类正常状态码,这些交给调用层按需求处理),通常我们只判断了手机的网络开关与模式,不易与发现其他网络异常。Retroift +RxJava轻松帮我们都拦截到,我们只需单独判断出来即可。
经过细致测试,按照由先到后,又子类到父类的策略拦截异常,保证能抓取到准确的异常信息
代码如下:

 /**
     * 协议层 读存数据发生错误才会走这里
     *
     * @param e
     */
    @Override
    public void onError (Throwable e) {
        //此处可拦截到的异常均发生在请求发起以后
        RxHttpLog.printI(TAG, "onError Throwable subClass :" + e.getClass() + " errorMsg:"+e.getMessage());
        if(e instanceof NullPointerException){
              /*
               * 响应返回或者缓存返回为null
               *此处的空指针来自于StringConverterFactory中转换响应结果时发生,会先调用onResponse 才会掉onError
               * 所以这里不下发给onFailure
               */
        }else if(e instanceof UnknownHostException){
            responseListener.onFailure(HttpCode.CODE_UNKNOW_HOST, "访问的目标主机不存在");

        }else if (e instanceof HttpException) {
            //SSL 验证成功 有正常的 响应返回,服务器无包装时 协议码为标准 401 404 等。。。
            HttpException httpException = (HttpException) e;
            responseListener.onFailure(httpException.code(), httpException.message());

        } else if (e instanceof SSLException) {
            //无法链接目标主机-本地网络无法访问外部服务器地址-服务器地址无法连接
            responseListener.onFailure(HttpCode.CODE_INTENET_IS_ABNORMAL, "无法与目标主机建立链接");

        } else if (e instanceof ConnectException
                || e instanceof SocketTimeoutException
                || e instanceof TimeoutException) {
            responseListener.onFailure(HttpCode.CODE_TIME_OUT, "链接超时");

        } else  if (e instanceof IOException) {
            //有网但请求失败包含:网络未授权访问外部地址,只读缓存时缓存区无缓存,等。。。
            responseListener.onFailure(HttpCode.CODE_RESPONSE_ERROR, "有网但请求失败包含:网络未授权访问外部地址,只读缓存时缓存区无缓存,等。。。");

        } else{

            responseListener.onFailure(HttpCode.CODE_UNKNOW, "未知网络错误");
        }
        responseListener.onFinish();
    }

END


框架已发布JitPack:
github开源地址:TBRetroift
欢迎Star me fork me
如果本篇内容有错或者在使用后有问题欢迎下方留言或者GitHub上提issue。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,384评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,845评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,148评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,640评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,731评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,712评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,703评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,473评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,915评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,227评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,384评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,063评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,706评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,302评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,531评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,321评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,248评论 2 352

推荐阅读更多精彩内容