QuickAF網(wǎng)絡(luò)連接及數(shù)據(jù)解析簡(jiǎn)介

QuickAF中使用Volley進(jìn)行網(wǎng)絡(luò)連接,使用Gson來解析響應(yīng)數(shù)據(jù)。為了更方便地執(zhí)行REST API網(wǎng)絡(luò)請(qǐng)求,QuickAF對(duì)Volley+Gson進(jìn)行了簡(jiǎn)單的封裝。

接口請(qǐng)求與響應(yīng)設(shè)計(jì)

接口

REST接口是基于HTTP協(xié)議的,一個(gè)接口的定義包含請(qǐng)求地址,請(qǐng)求方法,請(qǐng)求參數(shù),響應(yīng)信息。請(qǐng)求地址為一個(gè)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為查詢字符串。請(qǐng)求方法有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);
}

請(qǐng)求

一般來說接口的請(qǐng)求比較簡(jiǎn)單,如果請(qǐng)求是application/json,將請(qǐng)求對(duì)象轉(zhuǎn)為json字符串即可。但是實(shí)際當(dāng)中,仍然有許多接口使用的還是application/x-www-form-urlencoded,這種方式簡(jiǎn)單,而且適用于網(wǎng)頁。QuickAF默認(rèn)以后者來提交http請(qǐng)求,并且支持以下兩種請(qǐng)求格式

  1. 鍵值對(duì),這也是早期使用最多的,通過Map來存儲(chǔ)請(qǐng)求參數(shù)。
  2. 對(duì)象,通過反射機(jī)制將對(duì)象的屬性及屬性值轉(zhuǎn)化對(duì)鍵值對(duì),具有很高的可擴(kuò)展性,一旦接口有變更,比如接口要求添加uuid參數(shù),可以非常方便的修改請(qǐng)求基類來滿足業(yè)務(wù)需求,QuickAF建議使用這種方式來封裝請(qǐng)求。

通常在REST API中包含appKey, secret, uuid等全局請(qǐng)求參數(shù),QuickAF的sample app中定義的請(qǐng)求基類如下:

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

具體的業(yè)務(wù)API只需要繼承BaseRequest,然后添加具體的業(yè)務(wù)請(qǐng)求參數(shù),比如注冊(cè)的請(qǐng)求

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

對(duì)于GET請(qǐng)求,將請(qǐng)求對(duì)象轉(zhuǎn)為查詢字符串附在url中,對(duì)于POST請(qǐng)求,則將請(qǐng)求對(duì)象寫入body中。

響應(yīng)

REST API接口響應(yīng)一般包含狀態(tài)碼(status),提示信息(message)及業(yè)務(wù)對(duì)象(data),需要經(jīng)過json工具將其轉(zhuǎn)為對(duì)象,這個(gè)對(duì)象我們姑且稱之為接口對(duì)象。偽代碼如下:

class MyResponse {
  int code;//業(yè)務(wù)響應(yīng)狀態(tài)碼
  String message;//業(yè)務(wù)響應(yīng)信息,比如投票失敗
  Object data;//業(yè)務(wù)響應(yīng)對(duì)象,比如登錄,返回的是一個(gè)User對(duì)象
}

其實(shí)業(yè)務(wù)模塊往往關(guān)心的只有業(yè)務(wù)對(duì)象(data),因?yàn)閷?duì)于業(yè)務(wù)操作不成功的處理,可以在基類中統(tǒng)一處理。在QuickAF中,將接口對(duì)象抽象為IBaseResponse接口。

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

請(qǐng)求任務(wù)

如果我們將請(qǐng)求視為輸入,響應(yīng)視為輸出,那么對(duì)于一次網(wǎng)絡(luò)請(qǐng)求,使用代碼實(shí)現(xiàn)的話,就是:

abstract class MyTask<Input,Output> {
  void onSuccess(Output output); // 業(yè)務(wù)請(qǐng)求成功
  void onError(RestError error);//業(yè)務(wù)請(qǐng)求失敗
  Url getUrl();//業(yè)務(wù)請(qǐng)求地址
}

QuickAF中,已經(jīng)實(shí)現(xiàn)了網(wǎng)絡(luò)請(qǐng)求與數(shù)據(jù)解析功能,所以對(duì)開發(fā)者來說,只需專注于業(yè)務(wù)接口,即:接口地址,請(qǐng)求對(duì)象,返回的業(yè)務(wù)對(duì)象(data)。業(yè)務(wù)請(qǐng)求成功,在相關(guān)的界面填充業(yè)務(wù)數(shù)據(jù)(data),請(qǐng)求失敗,給出相應(yīng)的錯(cuò)誤信息(message)。

QuickAF有兩個(gè)執(zhí)行任務(wù)的方法

  1. 如果輸出為對(duì)象(Output)是一個(gè)對(duì)象,則需調(diào)用load方法,將Output的class傳進(jìn)去。
  2. 如果輸出為集合(List),則需調(diào)用load2List方法,將集合中的元素class傳進(jìn)去。
  3. 自動(dòng)識(shí)別,調(diào)用load方法,無需傳入Output的class。
// method 1
class MyTask<Object, User> {
  //... 獲取單個(gè)用戶,輸入為object,輸出為User
}
// 執(zhí)行任務(wù)
new MyTask().load(/*request*/null, User.class, /*don't load cache*/false);

// method 2 for List
class MyListTask<Object, List<User>> {
  //... 獲取用戶列表,輸入為object, 輸出為L(zhǎng)ist<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)

進(jìn)階操作

配置接口對(duì)象

接口對(duì)象,一個(gè)app,一般只有一個(gè)。定義如下

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

    /**
     * 狀態(tài)碼
     */
    public int code;

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

    /**
     * 數(shù)據(jù),業(yè)務(wù)對(duì)象
     */
    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的同學(xué),還可配置網(wǎng)絡(luò)連接使用OkHttp,需要寫一個(gè)OkHttpStack繼承自Volley的HurlStack,參考QuickAF示例app中的OkHttpStack.java.sample。

接口統(tǒng)一處理

主要是根據(jù)接口業(yè)務(wù)狀態(tài)碼進(jìn)行處理。比如定義業(yè)務(wù)操作成功,響應(yīng)碼為0,那么不為0的時(shí)候,就不應(yīng)該解析業(yè)務(wù)對(duì)象,轉(zhuǎn)入錯(cuò)誤分支。

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));
    }
}

數(shù)據(jù)模擬

接口對(duì)象或業(yè)務(wù)對(duì)象類需在mock()方法中給對(duì)象填充模擬值,參考示例工程中BaseInfo.java(這個(gè)類是所有業(yè)務(wù)對(duì)象模型的基類)。

最佳實(shí)踐

  • 所有的請(qǐng)求繼承一個(gè)BaseRequest,接口定義的全局請(qǐng)求參數(shù)在BaseRequest中定義
  • 一套接口API,定義一個(gè)全局的AppController及AppBaseTask來處理公共的業(yè)務(wù),比如業(yè)務(wù)攔截。
  • 所有的業(yè)務(wù)模型繼承一個(gè)BaseInfo
  • 一個(gè)Controller對(duì)應(yīng)一個(gè)界面,應(yīng)繼承AppController,包含若干網(wǎng)絡(luò)請(qǐng)求Task
  • 網(wǎng)絡(luò)請(qǐng)求Task回調(diào)作為內(nèi)部interface定義在具體的Controller中。

更多請(qǐng)參考demo app工程

關(guān)于

QuickAF是一個(gè)Android平臺(tái)上的app快速開發(fā)框架,歡迎讀者在github star或fork。本人寫作水平有限,歡迎廣大讀者指正,如有問題,可與我直接聯(lián)系或在我的官方博客中給出評(píng)論。

參考

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

本文永久鏈接: http://www.ieclipse.cn/2017/05/12/Android/quickaf-rest/ 未經(jīng)允許,禁止轉(zhuǎn)載,如有問題,請(qǐng)?jiān)谖业牟┛驮柬撁嫣峤辉u(píng)論。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 227,967評(píng)論 6 531
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,273評(píng)論 3 415
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 175,870評(píng)論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我,道長(zhǎng),這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,742評(píng)論 1 309
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 71,527評(píng)論 6 407
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,010評(píng)論 1 322
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,108評(píng)論 3 440
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,250評(píng)論 0 288
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,769評(píng)論 1 333
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 40,656評(píng)論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 42,853評(píng)論 1 369
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,371評(píng)論 5 358
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,103評(píng)論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,472評(píng)論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,717評(píng)論 1 281
  • 我被黑心中介騙來泰國(guó)打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 51,487評(píng)論 3 390
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 47,815評(píng)論 2 372

推薦閱讀更多精彩內(nèi)容

  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn),斷路器,智...
    卡卡羅2017閱讀 134,781評(píng)論 18 139
  • 國(guó)家電網(wǎng)公司企業(yè)標(biāo)準(zhǔn)(Q/GDW)- 面向?qū)ο蟮挠秒娦畔?shù)據(jù)交換協(xié)議 - 報(bào)批稿:20170802 前言: 排版 ...
    庭說閱讀 11,047評(píng)論 6 13
  • Android 自定義View的各種姿勢(shì)1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,613評(píng)論 25 708
  • 一說到REST,我想大家的第一反應(yīng)就是“啊,就是那種前后臺(tái)通信方式。”但是在要求詳細(xì)講述它所提出的各個(gè)約束,以及如...
    時(shí)待吾閱讀 3,444評(píng)論 0 19
  • 經(jīng)常有聽說狡猾煩人的老鼠會(huì)爬到汽車?yán)镆嗑€路,之前不明白為什么?這次,我的特斯拉停放在車庫(kù)里數(shù)月,居然也出現(xiàn)了同樣...
    Judy_c507閱讀 995評(píng)論 0 3