最近項目上Swift, 所以沒有采用AFN作網絡框架來使用, 即便可以使用OC轉Swift利用 AFN,但個人覺得 得學點新東西,要不項目就白做了. 所以使用AFN同一團隊開發的 Alamofire進行Swift開發.
Alamofire 使用:
- 和AFN一樣 作者并不希望我們直接使用 本類來操作, 建議建立網絡工具類, 繼承于它,然后設置單例, 這樣, 使用單例也就是使用 AFN框架. 這種方式便于我們做后期的修改
對于無論AFN和Alamofire設置單例都很簡單, 應為你本身調用的manager 就是一個單例, 所以你只需簡單的做一層包裝,就是調用了單例
這里我們 使用 static關鍵字, 全局量,只產生一份內存
// 單例
static let shareInstance: NetworkTools = {
let configuration = NSURLSessionConfiguration.defaultSessionConfiguration()
configuration.HTTPAdditionalHeaders = Manager.defaultHTTPHeaders
return NetworkTools(configuration: configuration)
}()
- 說真的我對于 Alamofire 就會使用一個方法,就是
request(...){...}
這里我就說一下使用,和AFN一個 對于網絡請求我們有 GET 和 POST 兩種請求方式. 但是對于 Alamofire這兩種方式我們只需要進行一個函數就可以實現了, Swift很支持函數的 重寫, 函數響應式編程思想,
這里我簡單的 將我寫的函數 進行貼出來, 各位看官 不要見笑
// 登陸狀態回調, 確定是否為已經正常登陸上了
func loginXQT(username name: String,
password code: String,
finish:(dict: [String: AnyObject]?, error: NSError?) -> ()){
SVProgressHUD.setMinimumDismissTimeInterval(1.0)
SVProgressHUD.showWithStatus("正在登陸")
SVProgressHUD.setDefaultMaskType(.Black)
// 1. 準備路徑
let path = "mUserLogin.htm"
// 2. 設置參數
let parameter = ["userLogingName": name , "userLogingCode": code]
/**
* 這里 需要學習的事, Alamofire 充分利用了 響應式 函數編程 原理. 將所有的 響應過程模塊化, 前者的響應 剛好是后者的 調用者. .... 這里我也不是 很好的理解 ,
request , 創建請求
responseJson ,對請求 進行 執行, 產生 數據交互
所有的 結果存儲在 response 中 . result. 結果集中存儲 成功與否, 數據有無
*/
request(.POST, baseURL + path, parameters: parameter).responseJSON { (response) -> Void in
if response.result.isSuccess
{
SVProgressHUD.showSuccessWithStatus("登陸成功")
finish(dict: response.result.value as? [String : AnyObject], error: nil)
}else{
SVProgressHUD.showErrorWithStatus("登陸失敗,請檢查網絡")
finish(dict: nil, error: response.result.error)
}
}
}
這里利用了函數響應式編程思想, 由request()函數
引出,響應responseJson{內部是個閉包}
, 這里就是簡單的思想, 前一個函數返回第二個函數的調用者, 然后直接調用,.
返回值 response 的類型是 Response<AnyObject, NSError>
一個結構體, 內部包含
public struct Response<Value, Error: ErrorType> {
// 關于響應的請求體
public let request: NSURLRequest?
// 響應體
public let response: NSHTTPURLResponse?
// 響應數據
public let data: NSData?
// 響應結果 --> 我們想要的數據
public let result: Result<Value, Error>
public init(request: NSURLRequest?, response: NSHTTPURLResponse?, data: NSData?, result: Result<Value, Error>) {
self.request = request
self.response = response
self.data = data
self.result = result
}
}
```objc
public enum Result<Value, Error: ErrorType> {
case Success(Value)
case Failure(Error)
/// Returns `true` if the result is a success, `false` otherwise.
// 文檔中指出, 如果確定 結果已成功請求, 則返回 true ,如果請求失敗則是 false
public var isSuccess: Bool {
switch self {
case .Success:
return true
case .Failure:
return false
}
}
/// Returns `true` if the result is a failure, `false` otherwise.
public var isFailure: Bool {
return !isSuccess
}
/// Returns the associated value if the result is a success, `nil` otherwise.
// 文檔中指出, 如果請求成功的話 , 就會返回 對應關聯的返回結果
// 這里結果我們所獲取的就是 對應的 數據庫 請求的 返回數據
public var value: Value? {
switch self {
case .Success(let value):
return value
case .Failure:
return nil
}
}
/// Returns the associated error value if the result is a failure, `nil`
otherwise.
// 文檔中指出, 請求失敗的話返回 error 的 錯誤嗎, 如果成功的話就是nil
public var error: Error? {
switch self {
case .Success:
return nil
case .Failure(let error):
return error
}
}
}
真的 大神就是大神,我可是寫不出這么的結構體寫法, 還是要多多學習,
對于結果體 我的理解就是,Swift已經很強大了, 結果體完全可以取代對象的建立, 就是創建屬性,分配內存什么的, 而且Swift 居然可以封裝初始化話代碼了
這里貼一段我的數據處理示例代碼
func getAllDepartmentRequest(deptName deptName: String?, finish:(dict: [String: AnyObject]?, error: NSError?) -> ()){
let path = "mGetAllDeptB.htm"
let parameter = ["deptName" : ""]
request(.POST, baseURL + path, parameters: parameter).responseJSON { (response) -> Void in
// 利用 result中的isSuccess 確定是否請求成功
if response.result.isSuccess{
SVProgressHUD.dismiss()
LXLLOG(response)
// 利用 result中的 value 來獲取響應結果數據
let date = response.result.value as! [String : AnyObject]
let source = date["callBackData"]
LXLLOG(source)
// 將數據傳入 閉包, 進行外界回調, 這里就要活用閉包, 進行數據回調,監聽
finish(dict: response.result.value as? [String : AnyObject], error: nil)
}else
{
SVProgressHUD.showErrorWithStatus("數據獲取失敗,請檢查網絡")
finish(dict: nil, error: response.result.error)
}
}
}