使用 protocol 優(yōu)雅處理 API

最近在遷移到 Swift3.0 過(guò)程中撞芍,為了逐步將 AFNetworking 轉(zhuǎn)移到 Alamofire 上,對(duì)于部分老的 OC 代碼順便一起做了重構(gòu)具垫,遂對(duì)于如何更好的組織 API 有了很多想法趟卸。

Why

在人員流動(dòng)與 Swift 的版本更新過(guò)程中,項(xiàng)目中逐漸有了以下對(duì)網(wǎng)絡(luò)操作的方式:

  • 基于 AFNetworking 的 OC 封裝
  • 基于 AFNetworking 的 Swift 封裝
  • 基于 Alamofire 的直接調(diào)用
  • 基于 Alamofire 的 Swift 封裝

于是乎峦筒,同一個(gè) API 可能有四種+的出現(xiàn)方式究西,加上同一個(gè)接口在不同服務(wù)中的調(diào)用,簡(jiǎn)直令人奔潰物喷。

可以復(fù)用的東西當(dāng)然要抽象出來(lái)卤材,But,How峦失?

How

Alamofire 4.0 版本中有很多變化: Request又進(jìn)一步添加了 4 個(gè)子類(lèi) DataRequest, DownloadRequest, UploadRequest 以及 StreamRequest扇丛,
pathmethod 對(duì)換了一個(gè)位置,ParameterEncoding 由枚舉變成了協(xié)議尉辑。于是乎原來(lái)基于 Alamofire 的直接調(diào)用的就全掛了帆精,修改的工程量浩大。所以隧魄,我選擇基于 Alamofire 的 Swift 封裝重新來(lái)組織 API卓练。

Swift 推薦你使用更多的值類(lèi)型,但有時(shí)候會(huì)沒(méi)有對(duì)象那么靈活购啄,基于我們的業(yè)務(wù)邏輯襟企,我選擇如下的 protocol ,具體的實(shí)現(xiàn)可以選擇

protocol APIType {
    var baseURL: String { get }
    var path: String { get }
    var url: String { get }
    var method: HTTPMethod { get }
    var parameters: Parameters { get }
    var encoding: ParameterEncoding { get }
    var headers: [String : String] { get } 
}

extension APIType {
    var url: String { return baseURL + path }
    var parameters: Parameters { return Parameters() }
    var encoding: ParameterEncoding { return URLEncoding() }
    var headers: [String : String] { return [:] }
}

由于 API 來(lái)自幾臺(tái)不同的服務(wù)器闸溃,同組的 API 有著相似的行為模式和錯(cuò)誤處理方式整吆,在這里定義好 baseURL, 同時(shí)也可以在這一層控制線(xiàn)上與線(xiàn)下環(huán)境,

protocol AnotherenAPIType: APIType { }

extension AnotherenAPIType {
    var baseURL: String {
        if EnvironmentManager.isOnline {
            return "https://release.anotheren.com"
        } else {
            return "https://debug.anotheren.com"
        }
    }
}

為了方便調(diào)用辉川,使用一個(gè)收集組來(lái)收集所有的同組 API

struct AnotherenAPI { }

extension AnotherenAPI {
    struct Login: AnotherenAPIType {
        let userName: String
        let password: String
        init(userName: String, password: String) {
            self.userName = userName
            self.password = password
        }
        var path: String { return "/login" }
        var method: HTTPMethod { return .post }
        var parameters: Parameters {
            return ["userName" : userName,
                    "password" : password]
        }
    }
}

Advance

實(shí)際上對(duì) API 返回?cái)?shù)據(jù)的處理也是固定的表蝙,比如這個(gè)接口是按照 JSON 格式返回?cái)?shù)據(jù)的,那其實(shí)就可以直接把數(shù)據(jù)處理后再返回乓旗,此處定義一個(gè) associatedtype ResultType 讓具體 API 定義時(shí)再確定 ResultType 的實(shí)際類(lèi)型

protocol JSONAPIType: APIType {
    associatedtype ResultType
    func handleJSON(json: JSON) -> Alamofire.Result<ResultType>
}

extension AnotherenAPI.Login: JSONAPIType {
    func handleJSON(json: JSON) -> Result<LoginInfo> {
        ...
    }
}

最后具體使用的時(shí)候就可以這樣處理府蛇,在回調(diào)的閉包中就可以直接拿到請(qǐng)求的結(jié)果 ResultType,類(lèi)似的屿愚,當(dāng)接口返回全是圖片的時(shí)候也可以再增加 PictureAPIType 協(xié)議汇跨,并讓相關(guān)接口實(shí)現(xiàn)即可务荆。

func request(_ api: APIType) -> DataRequest {
    let fullParameters = appendCustom(parameters: api.parameters)
    let fullHeaders = appendCustom(headers: api.headers)
    return Alamofire.request(api.url, method: api.method, parameters: fullParameters, encoding: api.encoding, headers: fullHeaders)
}

func requestJSON<T: JSONAPIType>(_ api: T, _ completionHandler: @escaping (Result<T.ResultType>) -> Void) -> DataRequest {
    return request(api).responseSwiftyJSON({ result in
        switch result {
        case .success(let json):
            completionHandler(api.handleJSON(json: json))
        case .failure(let error):
            completionHandler(Result.failure(error))
        }
    })
}

實(shí)際上,當(dāng)把 API 層這樣獨(dú)立拆開(kāi)后穷遂,測(cè)試起來(lái)也是非常方便的函匕。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市蚪黑,隨后出現(xiàn)的幾起案子盅惜,更是在濱河造成了極大的恐慌,老刑警劉巖忌穿,帶你破解...
    沈念sama閱讀 222,000評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件抒寂,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡掠剑,警方通過(guò)查閱死者的電腦和手機(jī)屈芜,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,745評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)朴译,“玉大人井佑,你說(shuō)我怎么就攤上這事《郑” “怎么了毅糟?”我有些...
    開(kāi)封第一講書(shū)人閱讀 168,561評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)澜公。 經(jīng)常有香客問(wèn)我姆另,道長(zhǎng),這世上最難降的妖魔是什么坟乾? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 59,782評(píng)論 1 298
  • 正文 為了忘掉前任迹辐,我火速辦了婚禮,結(jié)果婚禮上甚侣,老公的妹妹穿的比我還像新娘明吩。我一直安慰自己,他們只是感情好殷费,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,798評(píng)論 6 397
  • 文/花漫 我一把揭開(kāi)白布印荔。 她就那樣靜靜地躺著,像睡著了一般详羡。 火紅的嫁衣襯著肌膚如雪仍律。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 52,394評(píng)論 1 310
  • 那天实柠,我揣著相機(jī)與錄音水泉,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛草则,可吹牛的內(nèi)容都是我干的钢拧。 我是一名探鬼主播,決...
    沈念sama閱讀 40,952評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼炕横,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼源内!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起份殿,我...
    開(kāi)封第一講書(shū)人閱讀 39,852評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤姿锭,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后伯铣,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 46,409評(píng)論 1 318
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡轮纫,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,483評(píng)論 3 341
  • 正文 我和宋清朗相戀三年腔寡,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片掌唾。...
    茶點(diǎn)故事閱讀 40,615評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡放前,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出糯彬,到底是詐尸還是另有隱情凭语,我是刑警寧澤,帶...
    沈念sama閱讀 36,303評(píng)論 5 350
  • 正文 年R本政府宣布撩扒,位于F島的核電站似扔,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏搓谆。R本人自食惡果不足惜炒辉,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,979評(píng)論 3 334
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望泉手。 院中可真熱鬧黔寇,春花似錦、人聲如沸斩萌。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 32,470評(píng)論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)颊郎。三九已至憋飞,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間袭艺,已是汗流浹背搀崭。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,571評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人瘤睹。 一個(gè)月前我還...
    沈念sama閱讀 49,041評(píng)論 3 377
  • 正文 我出身青樓升敲,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親轰传。 傳聞我的和親對(duì)象是個(gè)殘疾皇子驴党,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,630評(píng)論 2 359

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

  • 嗯哼嗯哼蹦擦擦~~~ 轉(zhuǎn)載自:https://github.com/Tim9Liu9/TimLiu-iOS 目錄 ...
    philiha閱讀 4,910評(píng)論 0 6
  • 發(fā)現(xiàn) 關(guān)注 消息 iOS 第三方庫(kù)、插件获茬、知名博客總結(jié) 作者大灰狼的小綿羊哥哥關(guān)注 2017.06.26 09:4...
    肇東周閱讀 12,121評(píng)論 4 61
  • 寫(xiě)一首讓人有共鳴的詩(shī)港庄,譜一首讓人動(dòng)容的曲子,詩(shī)人還是歌者恕曲,一生只有一次鹏氧,也可以無(wú)悔了∨逡ィ“近鄉(xiāng)情更怯把还,不敢問(wèn)來(lái)人”,...
    不關(guān)燈閱讀 179評(píng)論 0 0
  • Y: 好久沒(méi)見(jiàn)到你了茸俭。 是從什么時(shí)候開(kāi)始習(xí)慣了你不在我身邊的日子吊履?可能是從學(xué)會(huì)熬夜開(kāi)始。 以前習(xí)慣早睡调鬓。因?yàn)槟悴幌?..
    葉云深閱讀 224評(píng)論 1 2
  • 晚上九點(diǎn)多鐘腾窝,家里的整個(gè)院子籠罩在黑的夜色里缀踪,鄉(xiāng)村大地已經(jīng)沉睡。在床上斜躺兩個(gè)多小時(shí)的我突然口渴虹脯,下床去客廳燒開(kāi)水...
    姬亞琳閱讀 379評(píng)論 0 1