# 01 Networking

实际应用中封装网络请求库 Webserver,目前只讨论 HTTP 请求:1.URL 拼接请求参数 -> 2.发起请求 -> 3.解析返回数据 -> 4.回调处理

final class Webservice {
    var url1:String = "xxxx"
    var url2:String = "xxxx"
    
    func loadEpisodes(completion: ([Episode]?) -> ()) {
        // 发起请求+解析+回调处理
    }

    func loadMedia(episode: Episode, completion: (Media?) -> ()) {
        // 发起请求+解析+回调处理
    }
}

Webservice 类现在和业务紧密联系在了一起,且随着请求增加会越来越臃肿;造成这一现象的原因有两点:1.请求链接(这个无法避免) 2. 解析(可能我想将NSData解析成JSON 可能是其他)。现在我们将这部分提取出来,构造一个 Resource ———— 为了更加通用,我们采用泛型:

struct Resource<A> {
    let url: URL
    let parse: (Data) -> A? 
}
let episodesResource = Resource<Data>(url: url, parse: { data in
    return data
})

Resource 和具体业务相关且独立,可复用,有点类似小时候游戏机的卡带,那么“小霸王”学习机加载口在哪里?

final class Webservice {
    func load<A>(_ resource: Resource<A>, completion: @escaping (A?) -> ()) {
        // 发起请求
        URLSession.shared.dataTask(with: resource.url as URL) { data, _, _ in
            let result = data.flatMap(resource.parse) 
            completion(result)
        }.resume()
    }
}

更多 flatMap 的使用,请见flatMap 温顾知新 —— 参照 Swift 源码实现讲解一文。

Data 返回数据,一般我们希望解析成JSON,XML,字典,甚至直接映射到某个类,例如:
// 1 Data -> Any

let resource1 = Resource<Any>(url: url, parse: { data in
     let json = try? JSONSerialization.jsonObject(with: data, options: [])
    return json
})

// 2 Data -> Dictionary

typealias JSONDictionary = [String: AnyObject]

let resource2 = Resource<[JSONDictionary]>(url: url, parse: { data in
    let json = try? JSONSerialization.jsonObject(with: data, options: [])
    return json as? [JSONDictionary]
})

// 3 Data -> [Episode]

struct Episode {
    let id: String
    let title: String
}

extension Episode {
    init?(dictionary: JSONDictionary) {
        guard let id = dictionary["id"] as? String,
            let title = dictionary["title"] as? String else { return nil }
        self.id = id
        self.title = title
    }
}

let episodesResource = Resource<[Episode]>(url: url, parse: { data in
    let json = try? JSONSerialization.jsonObject(with: data, options: [])
    guard let dictionaries = json as? [JSONDictionary] else { return nil }
    return dictionaries.flatMap(Episode.init)
})

对于 Resource 来说,输入为 Data 类型,而目的类型各式各样,如果让用户直接操作 Data 可能会不知所措,为此我们希望给相对友好点的类型 AnyObject,因此 Data->AnyObject 这一步转换我们会默认实现:

extension Resource {
    init(url: NSURL, parseJSON: AnyObject -> A?) {
        self.url = url
        // 默认解析闭包
        self.parse = { data in
            // json 类型为 AnyObject
            let json = try? NSJSONSerialization.JSONObjectWithData(data, options: [])
            
            return json.flatMap(parseJSON) // 注意这里parseJSON接受参数类型是 AnyObject
        }
    }
}

let episodesResource = Resource<[Episode]>(url: url, parseJSON: { json in
    guard let dictionaries = json as? [JSONDictionary] else { return nil }
    return dictionaries.flatMap(Episode.init)
})

ResourceEpisode 的资源之一,所以定义在 Episode 中比较合适:

extension Episode {
    var media: Resource<Media> {
        let url = NSURL(string: "http://localhost:8000/episodes/\(id).json")!
        // TODO Return the resource ...
    }
}

欢迎关注我的微博:@Ninth_Day

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

推荐阅读更多精彩内容