QuickAF网络连接及数据解析简介

QuickAF中使用Volley进行网络连接,使用Gson来解析响应数据。为了更方便地执行REST API网络请求,QuickAF对Volley+Gson进行了简单的封装。

接口请求与响应设计

接口

REST接口是基于HTTP协议的,一个接口的定义包含请求地址,请求方法,请求参数,响应信息。请求地址为一个URL,由基地址和接口路径和查询字符串组成。比如http://127.0.0.1:8080/meituan/api/1.0/user/login?token=xxxxxxx; http://127.0.0.1:8080/meituan/api/1.0/ 为基地址,一套api,其基地址是相同的。1.0为接口版本,user/login为接口路径,token=xxxxxx为查询字符串。请求方法有GET/POST/PUT/DELETE等。QuickAF将接口地址抽象为IUrl接口

public interface IUrl {
    /**
     * return http method, see {@link com.android.volley.Request.Method}
     *
     * @return http method
     * @see com.android.volley.Request.Method
     */
    int getMethod();

    /**
     * return the full url
     *
     * @return
     */
    String getUrl();

    /**
     * set query string to url.
     *
     * @param query query string, it a request parameter of string format usually
     */
    void setQuery(String query);
}

请求

一般来说接口的请求比较简单,如果请求是application/json,将请求对象转为json字符串即可。但是实际当中,仍然有许多接口使用的还是application/x-www-form-urlencoded,这种方式简单,而且适用于网页。QuickAF默认以后者来提交http请求,并且支持以下两种请求格式

  1. 键值对,这也是早期使用最多的,通过Map来存储请求参数。
  2. 对象,通过反射机制将对象的属性及属性值转化对键值对,具有很高的可扩展性,一旦接口有变更,比如接口要求添加uuid参数,可以非常方便的修改请求基类来满足业务需求,QuickAF建议使用这种方式来封装请求。

通常在REST API中包含appKey, secret, uuid等全局请求参数,QuickAF的sample app中定义的请求基类如下:

public class BaseRequest implements java.io.Serializable {
    public String appKey;
    public String secret;
    public String version;
    public String uuid;
}

具体的业务API只需要继承BaseRequest,然后添加具体的业务请求参数,比如注册的请求

public class RegisterRequest extends BaseRequest {
    public String phone;
    public String code;
    public String password;
}

对于GET请求,将请求对象转为查询字符串附在url中,对于POST请求,则将请求对象写入body中。

响应

REST API接口响应一般包含状态码(status),提示信息(message)及业务对象(data),需要经过json工具将其转为对象,这个对象我们姑且称之为接口对象。伪代码如下:

class MyResponse {
  int code;//业务响应状态码
  String message;//业务响应信息,比如投票失败
  Object data;//业务响应对象,比如登录,返回的是一个User对象
}

其实业务模块往往关心的只有业务对象(data),因为对于业务操作不成功的处理,可以在基类中统一处理。在QuickAF中,将接口对象抽象为IBaseResponse接口。

public interface IBaseResponse<Output> extends java.io.Serializable{
    /**
     * Return the business data object
     *
     * @return concrete business data
     */
    Output getData();
}

请求任务

如果我们将请求视为输入,响应视为输出,那么对于一次网络请求,使用代码实现的话,就是:

abstract class MyTask<Input,Output> {
  void onSuccess(Output output); // 业务请求成功
  void onError(RestError error);//业务请求失败
  Url getUrl();//业务请求地址
}

QuickAF中,已经实现了网络请求与数据解析功能,所以对开发者来说,只需专注于业务接口,即:接口地址,请求对象,返回的业务对象(data)。业务请求成功,在相关的界面填充业务数据(data),请求失败,给出相应的错误信息(message)。

QuickAF有两个执行任务的方法

  1. 如果输出为对象(Output)是一个对象,则需调用load方法,将Output的class传进去。
  2. 如果输出为集合(List),则需调用load2List方法,将集合中的元素class传进去。
  3. 自动识别,调用load方法,无需传入Output的class。
// method 1
class MyTask<Object, User> {
  //... 获取单个用户,输入为object,输出为User
}
// 执行任务
new MyTask().load(/*request*/null, User.class, /*don't load cache*/false);

// method 2 for List
class MyListTask<Object, List<User>> {
  //... 获取用户列表,输入为object, 输出为List<User>集合
}
new MyListTask().load2List(null, User.class, /*use cache*/true);

// method 3, auto load
new MyTask().load(null, false);// sampe to load(null, User.class, false)
new MyListTask().load(null, true);// sampe to load2List(null, User.class, false)

进阶操作

配置接口对象

接口对象,一个app,一般只有一个。定义如下

public class BaseResponse<Output> implements IBaseResponse {
    private static final long serialVersionUID = -3440061414071692254L;

    /**
     * 状态码
     */
    public int code;

    /**
     * 消息
     */
    public String message;

    /**
     * 数据,业务对象
     */
    public Output data;
    public Output getData() {
      return data;
    }
}

然后可以在Application.onCreate()中配置。

// typically, you just config volley in Application.onCreate
VolleyConfig config = new VolleyConfig.Builder()
    .setBaseResponseClass(WeatherBaseResponse.class)
    .build();
VolleyManager.init(getApplicationContext(), config);

如果有喜欢使用OkHttp的同学,还可配置网络连接使用OkHttp,需要写一个OkHttpStack继承自Volley的HurlStack,参考QuickAF示例app中的OkHttpStack.java.sample。

接口统一处理

主要是根据接口业务状态码进行处理。比如定义业务操作成功,响应码为0,那么不为0的时候,就不应该解析业务对象,转入错误分支。

protected abstract class AppBaseTask<Input, Output> extends RequestObjectTask<Input, Output> {

    @Override
    public boolean onInterceptor(IBaseResponse response) throws Exception {
        if (response instanceof BaseResponse) {
            BaseResponse resp = (BaseResponse) response;
            if (0 != resp.code) {
                onLogicError(new LogicError(null, resp.code, resp.message));
                throw new InterceptorError();
            }
        }
        return false;
    }

    public void onLogicError(LogicError error) {
        if (404 == error.getCode() || 104 == error.getCode()) { {
            // LoginActivity.go(MyApplication.instance);
            return;
        }
        onError(new RestError(error));
    }
}

数据模拟

接口对象或业务对象类需在mock()方法中给对象填充模拟值,参考示例工程中BaseInfo.java(这个类是所有业务对象模型的基类)。

最佳实践

  • 所有的请求继承一个BaseRequest,接口定义的全局请求参数在BaseRequest中定义
  • 一套接口API,定义一个全局的AppController及AppBaseTask来处理公共的业务,比如业务拦截。
  • 所有的业务模型继承一个BaseInfo
  • 一个Controller对应一个界面,应继承AppController,包含若干网络请求Task
  • 网络请求Task回调作为内部interface定义在具体的Controller中。

更多请参考demo app工程

关于

QuickAF是一个Android平台上的app快速开发框架,欢迎读者在github star或fork。本人写作水平有限,欢迎广大读者指正,如有问题,可与我直接联系或在我的官方博客中给出评论。

参考

QuickAF: https://github.com/Jamling/QuickAF

本文永久链接: http://www.ieclipse.cn/2017/05/12/Android/quickaf-rest/ 未经允许,禁止转载,如有问题,请在我的博客原始页面提交评论。

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

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,654评论 18 139
  • 国家电网公司企业标准(Q/GDW)- 面向对象的用电信息数据交换协议 - 报批稿:20170802 前言: 排版 ...
    庭说阅读 10,966评论 6 13
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,098评论 25 707
  • 一说到REST,我想大家的第一反应就是“啊,就是那种前后台通信方式。”但是在要求详细讲述它所提出的各个约束,以及如...
    时待吾阅读 3,423评论 0 19
  • 经常有听说狡猾烦人的老鼠会爬到汽车里咬断线路,之前不明白为什么?这次,我的特斯拉停放在车库里数月,居然也出现了同样...
    Judy_c507阅读 977评论 0 3