深入RxEasyHttp网络库教你3分钟学会自定义数据结构

github源码地址:https://github.com/zhou-you/RxEasyHttp

RxEasyHttp库默认ApiResult简介

在自定义ApiResult之前,先了解本库中默认提供的ApiResult方式,包含code、msg、data三个字段,其中默认code为0表示解析成功,内部是用ApiResult进行数据解析的。
默认ApiResult如下:

public class ApiResult<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() {
    //请求成功的判断方法,例如如果指定了code==0是成功,才会回调onSuccess(T t),那么code的非0其它数值都会走失败onError(ApiException e)
        return code == 0 ? true : false;
    }
}

返回正确的json数据:



返回错误的json数据:


code:服务器返回的状态码
msg:服务器返回的成功/失败信息
data:返回的真实数据,也是上层在页面需要展示的信息
isOk():请求成功的判断方法,例如指定了code==0是成功才会回调onSuccess(T t),那么code的其它数值都会走失败onError(ApiException e)

为什么需要自定义ApiResult

每个公司后台都有自己的一套数据结构规范,数据规范基本都会包含code码,错误信息、返回的数据data。但是他们的字段不一定是code、msg、data这样的三个字段。例如你的结构是error_code(code)、reason(msg)、result(data),那这样就会导致解析失败。而又不想因为几个字段的数据结构不一样,所有的解析和业务逻辑重新写一遍。而本库就想让你通过简单的改造,快速的接入,满足你的数据结构解析,所以提出了自定义数据结构的概念。

注:如果服务器人员在写接口时比较随意完全没有规范或者没有code、msg这样的结构,直接就是个bean,本库就不支持后期可以考虑支持,暂时没有提供针对非ApiResult结构的支持

自定义ApiResult原理

ApiResult类中包含code、msg、data的get和set方法,以覆盖重写的方式来解决,在自定义之前需要先用自己的数据结构对比库中默认的数据结构,一样的不用重复添加或者覆写。最主要的是覆写getCode()、getMsg()、getData()、isOk()四个方法,但不是这四个方法都必须全部覆写看看不一致的地方,哪些不一致就覆写哪些,set相关方法不用覆写。主要解决思想就是把不一致的数据结构,改造成和库中默认的结构一致或者靠拢。

注意:自定义ApiResult类时TestApiResult<T> extends ApiResult<T> 其中<T>必须写

举列1:服务器返回的结构是和默认的标准结构一模一样,只是code码为1表示成功,而库中0表示成功。那么只需要覆写isOk()

public class TestApiResult<T> extends ApiResult<T> {
    @Override
    public boolean isOk() {
        return getCode()==1;//code等于1表示成功,所有覆写isOk()
    }
}

举列2:服务器返回的结构是error_code、reason、result和默认的标准结构完全不一样

public class TestApiResult<T> extends ApiResult<T> {
    int error_code; //对应默认标准ApiResult的code
    String reason;//对应默认标准ApiResult的msg
    T result;//对应默认标准ApiResult的data

    @Override
    public T getData() {//因为库里使用data字段,而你的结构是result,所以覆写getData()方法,setData()方法不用覆写
        return result;
    }

    @Override
    public String getMsg() {//因为库里使用msg字段,而你的结构是reason,所以覆写getMsg()方法,setMsg()方法不用覆写
        return reason;
    }

    @Override
    public int getCode() {//因为库里使用code字段,而你的结构是error_code,所以覆写getCode()方法,setCode()方法不用覆写
        return error_code;
    }

    @Override
    public boolean isOk() {//因为库里使用code字段,code==0表示成功,但是你的数据结构是error_code==0,所以覆写isOk(),用error_code==0做为判断成功的条件
        return error_code == 0;//表示成功
    }
}

举列3:服务器返回的结构是no,msg,obj结构,其中msg是一样的就不用再特殊处理,no、obj不一样需要处理


注:其它更多情景自己举一反三

自定义ApiResult请求网络步骤

假如你的请求链接是:
http://japi.juhe.cn/joke/content/list.from?key=f5236a9fb8fc75fac0a4d9b8c27a4e90&page=1&pagesize=10&sort=asc&time=1418745237

第一步:分析数据结构

如果返回的字符串不好查看结构,可以使用工具分析,例如json格式化工具sojsonbejson、Chrome浏览器插件JSON-Handle、Json Viewer等
链接返回的json数据:

第二步:自定义数据结构

看到上面返回的json格式,结合自定义ApiResult原理,定义结构如下:

public class TestApiResult<T> extends ApiResult<T> {
    int error_code; //对应默认标准ApiResult的code
    String reason;//对应默认标准ApiResult的msg
    T result;//对应默认标准ApiResult的data
    @Override
    public T getData() {
        return result;
    }
    @Override
    public String getMsg() {
        return reason;
    }
    @Override
    public int getCode() {
        return error_code;
    }
    @Override
    public boolean isOk() {
        return error_code == 0;//表示成功
    }
}

注:定义出的TestApiResult<T>属于公共的,同样的数据结构都可以使用到

第三步:准备数据解析的result

自定义数据结构TestApiResult是个通用的结构,实体的data是result,result是泛型T,具体表示什么,需要你自己查看返回json结构,本例中result字段中包含一个data字段,data字段是集合,所以如下:

public class Data {
    private String content;
    private String hashId;
    private int unixtime;
    private String updatetime;
    public String getContent() {
        return content;
    }
    public Data setContent(String content) {
        this.content = content;
        return this;
    }
    public String getHashId() {
        return hashId;
    }

    public Data setHashId(String hashId) {
        this.hashId = hashId;
        return this;
    }
    public int getUnixtime() {
        return unixtime;
    }
    public Data setUnixtime(int unixtime) {
        this.unixtime = unixtime;
        return this;
    }
    public String getUpdatetime() {
        return updatetime;
    }

    public Data setUpdatetime(String updatetime) {
        this.updatetime = updatetime;
        return this;
    }

    @Override
    public String toString() {
        return "Data{" +
                "content='" + content + '\'' +
                ", hashId='" + hashId + '\'' +
                ", unixtime=" + unixtime +
                ", updatetime='" + updatetime + '\'' +
                '}';
    }
}
public class Result {
    private List<Data> data;
    public void setData(List<Data> data) {
        this.data = data;
    }
    public List<Data> getData() {
        return data;
    }

    @Override
    public String toString() {
        return "Result{" +
                "data=" + data +
                '}';
    }
}

第四步:网络请求数据

自定义的网络请求必须用回调代理execute(CallBackProxy<? extends ApiResult<T>, T> proxy)来请求网络

EasyHttp.get("http://japi.juhe.cn/joke/content/list.from")
                .params("key", "f5236a9fb8fc75fac0a4d9b8c27a4e90")
                .params("page", "1")
                .params("pagesize", "10")
                .params("sort", "asc")
                .params("time", "1418745237")
                .execute(new CallBackProxy<TestApiResult<Result>, Result>(new ProgressDialogCallBack<Result>(mProgressDialog) {
                    @Override
                    public void onError(ApiException e) {
                        super.onError(e);
                        showToast(e.getMessage());
                    }

                    @Override
                    public void onSuccess(Result result) {
                        if (result != null) showToast(result.toString());
                    }
                }) {
                });

注:通过以上四个步骤就完成了自定义数据结构网络请求数据,是不是觉得写法会有点长,CallBackProxy的泛型参数每次都需要填写,其中TestApiResult是继承ApiResult的,对于一个实际项目来讲后台的服务器数据结构是固定的,也就是TestApiResult是固定的,这种方法太麻烦了,那有没有简单的写法,而又不破坏库的使用方式,请看自定义ApiResult高级用法。

自定义ApiResult高级用法

上面已经提到过使用自定义网络请求必须要用new CallBackProxy<TestApiResult<Result>, Result>,会有很多重复的操作,结合上面我们看下如何简写。讲解之前先说下库中实现方式,网络请求EasyHttp.get()是返回一个GetRequest请求,EasyHttp.post()是返回一个PostRequest请求.如果想让所有的请求execute()方法都自动加上new CallBackProxy<TestApiResult<T>, T>,就需要修改execute()方法,而execute()方法是在GetRequest和PostRequest中,所以我们需要分别继承GetRequest和PostRequest,重写execute()方法。具体操作如下:

自定义GetRequest请求

注:下面代码可以直接拷贝过去使用,其中的TestApiResult修改成自己的ApiResult就可以了

public class CustomGetRequest extends GetRequest {
    public CustomGetRequest(String url) {
        super(url);
    }

    @Override
    public <T> Observable<T> execute(Type type) {
        return super.execute(new CallClazzProxy<TestApiResult<T>, T>(type) {
        });
    }

    @Override
    public <T> Observable<T> execute(Class<T> clazz) {
        return super.execute(new CallClazzProxy<TestApiResult<T>, T>(clazz) {
        });
    }

    @Override
    public <T> Subscription execute(CallBack<T> callBack) {
        return super.execute(new CallBackProxy<TestApiResult<T>, T>(callBack) {
        });
    }
}

自定义PostRequest请求

注:下面代码可以直接拷贝过去使用,其中的TestApiResult修改成自己的ApiResult就可以了

public class CustomPostRequest extends PostRequest {
    public CustomPostRequest(String url) {
        super(url);
    }

    @Override
    public <T> Observable<T> execute(Type type) {
        return super.execute(new CallClazzProxy<TestApiResult<T>, T>(type) {
        });
    }

    @Override
    public <T> Observable<T> execute(Class<T> clazz) {
        return super.execute(new CallClazzProxy<TestApiResult<T>, T>(clazz) {
        });
    }

    @Override
    public <T> Subscription execute(CallBack<T> callBack) {
        return super.execute(new CallBackProxy<TestApiResult<T>, T>(callBack) {
        });
    }
}

新建请求的管理类方便使用

public class HttpManager {
    /**
     * get请求
     */
    public static GetRequest get(String url) {
        return new CustomGetRequest(url);
    }

    /**
     * post请求
     */
    public static PostRequest post(String url) {
        return new CustomPostRequest(url);
    }
}
新的网络请求方式
HttpManager.get("http://japi.juhe.cn/joke/content/list.from")
                .params("key", "f5236a9fb8fc75fac0a4d9b8c27a4e90")
                .params("page", "1")
                .params("pagesize", "10")
                .params("sort", "asc")
                .params("time", "1418745237")
                .execute(new ProgressDialogCallBack<Result>(mProgressDialog) {//这么实现是不是没有代理了
                    @Override
                    public void onError(ApiException e) {
                        super.onError(e);
                        showToast(e.getMessage());
                    }

                    @Override
                    public void onSuccess(Result result) {
                        if (result != null) showToast(result.toString());
                    }
                });

使用方式对比

具体源码请查看GitHub中提供的Demo:https://github.com/zhou-you/RxEasyHttp

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

推荐阅读更多精彩内容