Retrofit使用教程:组合RxJava简易封装使用

本文将围绕Retrofit的组装请求管理器、发起请求、请求响应的封装进行介绍。
以获取手机号码归属地的整个流程为例:

Get请求

请求API类,这个类等待Retrofit通过动态代理的方式将这个接口的方法以及对应得注解生成一个http请求,在把这个http请求交给OkHttp处理。

public interface HttpService {
    @GET(GlobalVar.NetPorts.WEATHER)
    Observable <HttpResult<PhoneLocalBean>> queryWeather(@QueryMap Map<String, String> options);
}

请求管理器

创建一个请求管理器用来配置Retrofit与OkHttp的基本属性。

public class HttpManager {

    public static final String SERVER = GlobalVar.SERVER; //服务器根地址
    private HttpService mHttpService;
    private Retrofit mAdapter;
    private static HttpManager instance;

    private HttpManager() {
        mAdapter = new Retrofit.Builder().baseUrl(SERVER).addConverterFactory(ScalarsConverterFactory.create()).addConverterFactory(GsonDConverterFactory.create()).addCallAdapterFactory(RxJavaCallAdapterFactory.create()).client(getBuilder().build()).build();
    }

    public static HttpManager getInstance() {
        if (instance == null) {
            instance = new HttpManager();
        }
        return instance;
    }

    public HttpService sendRequest() {
        if (mHttpService == null) {
            mHttpService = mAdapter.create(HttpService.class);
        }
        return mHttpService;
    }

    private OkHttpClient.Builder getBuilder() {
        //这里的存储位置只是简单获取,根据实际需要修改
        File cacheFile = new File(BaseApplication.getContext().getCacheDir().getAbsolutePath(), "ShopHttpCache");
        Cache cache = new Cache(cacheFile, 1024 * 1024 * 100);

        OkHttpClient.Builder builder = new OkHttpClient.Builder();
        builder.addInterceptor(new HttpCacheInterceptor());
        builder.cache(cache);
        builder.readTimeout(GlobalVar.READ_TIMEOUT, TimeUnit.SECONDS);
        builder.connectTimeout(GlobalVar.CONNECT_TIMEOUT, TimeUnit.SECONDS);
        builder.writeTimeout(GlobalVar.WRITE_TIMEOUT, TimeUnit.SECONDS);
        builder.retryOnConnectionFailure(true);
        return builder;
    }
}

请求拦截器

用于对请求的发起与响应的拦截,这里我们可以增加统一的Header,客户端与服务端的加密验证也可以放在这里,网络缓存等等。

public class HttpCacheInterceptor implements Interceptor {

    private static final String TAG = "HttpManager";

    @Override public Response intercept(Chain chain) throws IOException {
        Request original = chain.request();
        Request.Builder requestBuilder = original.newBuilder().method(original.method(), original.body());

        Headers.Builder hb = new Headers.Builder();
        addHeader(hb);
        if (!NetUtils.isConnected(BaseApplication.getContext())) {
            //网络不可用
            requestBuilder.cacheControl(CacheControl.FORCE_CACHE);
        } else { 
            requestBuilder.cacheControl(CacheControl.FORCE_NETWORK);
        }
        Request request = requestBuilder.headers(hb.build()).build();
        Log.d(TAG, "地址:" + request.url());

        try {
            Response response = chain.proceed(request);
            String cookie = response.headers().get("Set-Cookie");

            if (NetUtils.isConnected(BaseApplication.getContext())) { //如果网络可用
                int maxAge = 60 * 3;
                response = response.newBuilder().removeHeader("Pragma").header("Cache-Control", "public, max-age=" + maxAge).build();
            } else {
                int maxStale = 60 * 60 * 24;
                response = response.newBuilder().removeHeader("Pragma").header("Cache-Control", "public, only-if-cached, max-stale=" + maxStale).build();
            }
            return response;
        } catch(Exception err) {
            Log.e("HttpManager", "http=============" + err.getLocalizedMessage());
        }
        return null;
    }

    private void addHeader(Headers.Builder header) {
        header.add("apikey", "8ca4b101096587f725ff69a07ff4d188");
    }
}

发起请求

这里的addMainSubscription用于简化书写并统一管理发起的订阅,当Activity退出时统一取消订阅。

@Override
public void queryWeather(String phone) {
    Map < String, String > options = new HashMap < >();
    options.put("phone", phone);
    addMainSubscription(HttpManager.getInstance().sendRequest().queryWeather(options), new HttpResultCallBack<PhoneLocalBean> () {

        @Override public void onResponse(PhoneLocalBean bean, int status) {
            if (bean != null) {
                mView.onUserLoadCompleted(bean);
            } else {
                mView.onUserLoadError();
            }
        }

        @Override public void onErr(String err, int status) {
            mView.showToast(err);
            mView.onUserLoadError();
        }
    });
}

自定义请求响应回调

对响应数据成功失败的封装,而一般的数据返回格式基本分为:
1、状态码
2、提示信息
3、具体数据内容
具体数据体可以是集合也可以是对象,所以使用了泛型HttpResult<M>统一处理。这里定义的M一方面是告诉GSON我的子数据格式是什么,另一方面用于回调方法中返回对应得数据类型。

public abstract class HttpResultCallBack<M> extends Subscriber <HttpResult<M>> {

    public abstract void onResponse(M m, int status);
    public abstract void onErr(String msg, int status);

    @Override public void onCompleted() {}

    @Override public void onError(Throwable e) {
        if (e != null) {
            if (e instanceof ResultException) {
                ResultException err = (ResultException) e;
                onErr(err.getErrMsg(), GlobalVar.RESULT_UNLOGIN);
            } else {
                onErr("网络异常,请检查网络", GlobalVar.RESULT_UNLOGIN);
                Log.d("HttpManager", "解析失败==:" + e.getMessage());
            }
        }
        onCompleted();
    }

    private void onHttpFail(String msg, int status) {
        onErr(msg, status);
    }

    @Override public void onNext(HttpResult < M > result) {
        String jsonResponse = new Gson().toJson(result);
        Log.d("HttpManager", "返回ok==:" + jsonResponse);
        if (result.getErrNum() == GlobalVar.RESULT_OK) {
            onResponse(result.getRetData(), GlobalVar.RESULT_OK);
        } else {
            onHttpFail(result.getErrMsg(), GlobalVar.RESULT_UNLOGIN);
        }
    }
}

结语

这里的代码都是结合具体项目来写的,虽然是抽出来的精简版但更容易读懂。我觉得直接看代码更容易理解一段代码的思想,所以大多都是以代码为主注释为辅。最后有哪些不足的地方也谢谢大家指出来~

github源码下载

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

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,448评论 25 707
  • 又是一年中秋佳节,祝各位中秋节快乐。 今天我们来聊聊这个最近很火的网络请求库retrofit,在此基础上会延伸出一...
    涅槃1992阅读 7,771评论 13 133
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,596评论 18 139
  • 整体Retrofit内容如下: 1、Retrofit解析1之前哨站——理解RESTful2、Retrofit解析2...
    隔壁老李头阅读 15,026评论 4 39
  • 传说这条小路 曾有一位女子走过 我想踏着她的印子 寻她的芳香 明月高悬 树影斑驳 路上只我一个 月光泻着我 如一位...
    马人言阅读 166评论 0 2