Retrofit+Rxjava+OKHttp3之Session过期处理

前言
在app实际请求服务器接口的过程中,经常会遇到session过期的情况,这时候需要我们重新登录刷新session。
期初实现的方案是,发现session过期需要登录时,直接跳转到登录界面,登陆后跳转首页。
但更合理的情形是一旦发现session过期,直接调用登录接口刷新session,之后继续原来的业务,即在此请求原来的业务接口。
本文就将针对使用OKHttp3的底层网路框架的情形,说一下如何实现session过期的统一处理。
有兴趣的同学可以加入学习小组QQ群: 193765960做进一步的讨论。

版权归作者所有,转发请注明出处:https://www.jianshu.com/u/d43d948bef39

1. 总体方案:拦截器

1)在网络请求的底层OKhttp层设置拦截器,拦截网络请求和响应数据。
2)分析响应数据,根据响应的状态判断是否发生session过期。
3)如果没有过期,则正常返回响应数据。
4)如果发生session过期,则取消当前响应数据,生成刷新session的请求(一般是登录请求)。
5)如果session刷新成功,根据原来业务的请求数据从新发起请求。
6)如果session刷新失败,则返回session过期的异常响应,在上层解析该响应后跳转登录界面。
7)从登录界面重新登陆后,跳转到首页。

2. 拦截器的实现

import okhttp3.Interceptor;
import okhttp3.Request;
import okhttp3.Response;

public class SessionKeyInterceptor implements Interceptor {
    @Override
    public Response intercept(Chain chain) throws IOException {
        Request old_request = chain.request();
        Response old_response = chain.proceed(old_request);
        //下面这行代码注意下,不要使用Gson等工具对old_response.body等直接转化,会出问题的。
        //RequestHelper: 工具类小伙伴们可以加群,单独找我索要
        JSONObject obj = RequestHelper.response2Object(old_response);
        //如果session过期,则重新登录获取sessionkey
        if (obj != null && TextUtils.equals(obj .optString("error_code"), MyConfig.getSessionKeyErrorNumber())) {

            //上层注入的请求MyRequest 
            MyRequest loginp = LoginUtils.getRequest();
            if(null == loginp || loginp.size()==0){
                return old_response;
            }

            if(null == LoginUtil.getLoginApi()){
                return originalResponse;
            }

            Request login_request = RequestHelper.buildGetRequest(loginparam, LoginUtils.getLoginApi());
            Response login_response = chain.proceed(login_request);
            if (login_response.isSuccessful()) {
                JSONObject obj2 = RequestHelper.response2Object(login_response);
                if(obj2!= null && (TextUtils.equals(obj2.optString("error_code"), "//api请求成功的状态码"))){
                    //登录成功,执行原始request
                    String sessionKey = "";
                    try {
                        sessionKey = obj2.getJSONObject("jsondata").getString("session_key");
                    } catch (JSONException e) {
                        e.printStackTrace();
                    }
                    login_response.body().close();
                    Map<String, Object> param = RequestHelper.parseParams(old_request);
                    //更新old_request的session_key等可变参数
                    param.put("sessionKey",sessionKey);
                    MyRequest request = new MyRequest ();
                    request.put(param);
                    originalRequest = RequestHelper.handler(request, old_request);
                    originalResponse.body().close();
                    return chain.proceed(old_request);
                }
            }
        }
        return old_response;
    }

基于项目代码安全的需要,对以上代码做了处理,不过大体的思路都是可用的

3. 登录请求的上层注入

为了更好地适用于不同的项目,和后期代码的维护,判断session是否过期的判断依据采用上层配置并注入的方式。
当然,您也可以直接在Okhttp层写死,开发难度会简单很多,只是后期维护或者做项目移植时需要特别注意修改。
发起session刷新的接口请求参数也是同样的处理。不再一一赘述。

4. 存在问题

长时间在后台静默APP,数据有可能会被gc掉,所以这种情况下,系统不会无缝的去成功刷新session并执行目标请求的业务逻辑,而是会直接启动登录界面。
这个拦截器,仅仅针对了session过期的情况,其实功能逻辑具有通用性,可以考虑和业务解耦,通过注入的方式支持更多异常的无缝处理。

以上。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,881评论 18 139
  • 从三月份找实习到现在,面了一些公司,挂了不少,但最终还是拿到小米、百度、阿里、京东、新浪、CVTE、乐视家的研发岗...
    时芥蓝阅读 42,360评论 11 349
  • iOS网络架构讨论梳理整理中。。。 其实如果没有APIManager这一层是没法使用delegate的,毕竟多个单...
    yhtang阅读 5,251评论 1 23
  • 一、选择你的出生 1、你有个很有钱的爹,但是你的身体很差但不至于死 2、你很穷但是身体很好,是一夜十次郎的那种 二...
    孤叶南飞阅读 842评论 0 4
  • 朋友在平安保险工作,托我帮忙给新产品写首诗宣传,不知道哪来的灵感,还真的写出来了,与大家分享! 保险的名字叫玺越人...
    天佑_f901阅读 175评论 0 1