相信寫過Swift的人應(yīng)該都知道Alamofire英岭,它是AFNetworking的Swift版本湿右,同一個(gè)作者寫的毅人。之前在項(xiàng)目中我也一直使用Alamofire,但是升級到Xcode7之后舊版的Alamofire不能用了划煮,最新版的又只支持iOS8之后的系統(tǒng)缔俄,而公司項(xiàng)目還得兼容iOS7,所以接下來不打算用它了蟹略。
我的需求比較簡單遏佣,只要能發(fā)送GET請求獲取數(shù)據(jù)以及發(fā)送POST請求提交數(shù)據(jù)就好了,大致看了一下Alamofire的源碼又上網(wǎng)查了點(diǎn)資料之后意敛,花了不到半天寫了幾個(gè)簡單的函數(shù)膛虫,項(xiàng)目又能正常跑起來了。
iOS7之后的系統(tǒng)都支持NSURLSession
,我們就把它稍微封裝一下好了碴倾。
func getDataFrom(urlString: String, method: HTTPMethod, parameter: [String: String]?, completionHandler: Callback) throws {
// let config = NSURLSessionConfiguration.defaultSessionConfiguration()
// config.timeoutIntervalForRequest = 20
// let session = NSURLSession(configuration: config)
guard let url = NSURL(string: urlString) else {
throw Error.InvalidURL
}
let session = NSURLSession.sharedSession()
let request = NSMutableURLRequest(URL: url)
request.HTTPMethod = method.rawValue
switch method {
case .POST:
//如果參數(shù)為nil或者字典中沒有元素,則拋出異常
guard let param = parameter else {
throw Error.NoParameter
}
guard param.isEmpty else {
throw Error.NoParameter
}
request.addValue("application/json", forHTTPHeaderField: "Content-Type")
request.addValue("application/json", forHTTPHeaderField: "Accept")
do {
request.HTTPBody = try NSJSONSerialization.dataWithJSONObject(param, options: [])
} catch {
print(error)
}
case .GET:
break
}
let task = session.dataTaskWithRequest(request) {data, response, error in
guard let result = data where error == nil else {
printLog("no data: \(error)")
return
}
completionHandler(data: result)
}
//啟動
task.resume()
}
這個(gè)函數(shù)聲明的時(shí)候在->
前加了一個(gè)throws
,表明這個(gè)函數(shù)是可以拋出異常的僧须。其實(shí)以往iOS開發(fā)比較推崇"Let it crash!"的哲學(xué),不過Swift一直很強(qiáng)調(diào)安全性示绊,Apple顯然也并不僅僅滿足于讓Swift困守iOS開發(fā)領(lǐng)域暂论,加上早就公布了年底要開源次乓,大家也很期待它作為一門通用編程語言在其他領(lǐng)域的作為寥茫。從各方面來看椎麦,Swift2.0增加了對異常處理的支持都在情理之中闻蛀。從此你的App就不能輕易的狗帶了~
我對異常處理的理解很淺薄觉痛,說實(shí)話平常自己也不怎么喜歡用。在我看來異常處理最重要的用途有兩點(diǎn):
- 寫底層框架的時(shí)候可以拋出一些異常讓框架的使用者去處理手蝎,這樣框架會顯得更加靈活盗尸。
- 保存錯(cuò)誤日志,便于查詢和調(diào)試鞍时。
像我上面那個(gè)函數(shù)扣蜻,如果純粹是自己用的話,其實(shí)我會選擇在出錯(cuò)的地方直接處理錯(cuò)誤或者打印錯(cuò)誤信息锐极,譬如把throw Error.InvalidURL
改成
printLog("Invalid URL")
return
這樣也省得調(diào)用函數(shù)的時(shí)候一堆try-catch
芳肌。當(dāng)然有些錯(cuò)誤當(dāng)前函數(shù)確實(shí)是處理不了肋层,那該拋還得拋栋猖。
上面那個(gè)函數(shù)還可以封裝一下汪榔,分成兩個(gè),一個(gè)用來發(fā)送 GET
請求接收JSON
數(shù)據(jù)雌团,一個(gè)用來POST
JSON
數(shù)據(jù)并接收返回信息士聪。具體如下:
func getJsonFrom(url: String, completion: (json: JSON) -> Void) {
do {
try getDataFrom(url, method: HTTPMethod.GET, parameter: nil) { data in
let json = JSON(data: data)
//主線程進(jìn)行UI操作
dispatch_sync(dispatch_get_main_queue()) {
completion(json: json)
}
}
} catch Error.InvalidURL {
printLog("GET: invalid url")
} catch {
printLog("Unknown error")
}
}
func postJson(dict: [String: String], toUrl url: String, completion: (json: JSON) -> Void) {
do {
try getDataFrom(url, method: HTTPMethod.POST, parameter: dict) { data in
let json = JSON(data: data)
//主線程進(jìn)行UI操作
dispatch_sync(dispatch_get_main_queue()) {
completion(json: json)
}
}
} catch Error.InvalidURL {
printLog("POST: invalid url")
} catch Error.NoParameter {
printLog("Parameter is empty")
} catch {
printLog("Unknown error")
}
}
完整代碼在這里戚嗅,里面還有一個(gè)圖片緩存的函數(shù)枢舶,有興趣的話可以看看。如果跟我有同樣需求的同學(xué)可以把HttpManager.swift
clone下來直接拖到項(xiàng)目里躏尉,建議配合SwiftyJSON(一個(gè)很好用的第三方JSON解析庫)使用后众,直接把Source文件夾里的SwiftyJSON.swift
這個(gè)文件也一起拖到項(xiàng)目中好了,要用Cocoapods導(dǎo)入framework的話似乎只能支持iOS8之后的系統(tǒng)了教藻。
對了還有一點(diǎn)右锨,我一開始用JSON(data: data)
來初始化JSON
數(shù)據(jù)的時(shí)候總是不成功,于是我看了下SwiftyJSON中JSON
這個(gè)struct
的構(gòu)造函數(shù)悄窃,它先調(diào)用了蘋果提供的class func JSONObjectWithData(_ data: NSData, options opt: NSJSONReadingOptions) throws -> AnyObject
函數(shù)蹂窖,然后把返回的AnyObject
對象賦值給自身屬性object
:
public init(data:NSData, options opt: NSJSONReadingOptions = .AllowFragments, error: NSErrorPointer = nil) {
do {
let object: AnyObject = try NSJSONSerialization.JSONObjectWithData(data, options: opt)
self.init(object)
} catch let aError as NSError {
if error != nil {
error.memory = aError
}
self.init(NSNull())
}
}
public init(_ object: AnyObject) {
self.object = object
}
opt
這個(gè)參數(shù)有三個(gè)可選值:MutableContainers
, MutableLeaves
, AllowFragments
,分別表示:
- 可以把數(shù)組或者字典轉(zhuǎn)化成可變對象瞬测;
- 可以把JSON對象樹中作為葉子節(jié)點(diǎn)的字符串轉(zhuǎn)化成可變字符串纠炮;
- 允許解析最外層對象不是
NSArray
或NSDictionary
實(shí)例的JSON
數(shù)據(jù)
三種我都試了灯蝴,都不行绽乔,最后我試了下[]
,也就是傳入一個(gè)空值折砸,居然行了睦授。于是我索性把JSON
的初始化函數(shù)改了:
public init(data:NSData, options opt: NSJSONReadingOptions = [], error: NSErrorPointer = nil) {
//其余不變
和我遇到相同問題的同學(xué)也可以這樣試試。有什么問題或指教歡迎評論怖辆。