本篇將從四個方面對iOS開發中經常使用到的AFNetworking框架進行講解:
一、什麼是 AFN
二、為什麼要使用 AFN
三、AFN 怎麼用
三、AFN和ASI的區別
AFN 全稱為 AFNetworking,是一個構建與在 NSURLConnection、NSOperation 以及其他熟悉的 Founation 技術之上的一個第三方網絡框架。
AFN 對網絡請求部分做了很好的封裝,並且擁有良好的架構,豐富的api,以及模塊化構建方式,使得使用起來非常輕松.。
AFN 主要使用在2個場景中:
一、發送網絡請求
二、實時監測網絡狀態
AFHTTPRequestOperationManager:內部封裝的是 NSUrlConnection,網絡請求管理類,用來負責發送網絡請求,是使用最多的一個類.
AFHTTPSessionManager :內部封裝的是 NSUrlSession,網絡請求管理類,用來負責發送網絡請求,是使用做多的一個類.
兩個網絡請求管理類定義的 API 完全相同:
[[AFHTTPRequestOperationManager manager] GET:nil parameters:nil success:^(AFHTTPRequestOperation *operation, id responseObject) { // } failure:^(AFHTTPRequestOperation *operation, NSError *error) { // }]; [[AFHTTPSessionManager manager] GET:nil parameters:nil success:^(NSURLSessionDataTask *task, id responseObject) { // } failure:^(NSURLSessionDataTask *task, NSError *error) { // }];
一些主要的工具類:
AFNetworkReachabilityManager.h :實時監測網絡狀態改變的管理類.
AFSecurityPolicy.h :HTTPS 需要使用.
AFURLRequestSerialization: 數據解析的時候會使用.
AFHTTPRequestSerializer: 萬能解析器/對服務器返回的數據不做任務處理.
AFJSONResponseSerializer: JSON解析器.
AFXMLParserResponseSerializer: XML解析器.
AFHTTPRequestOperationManager 對NSURLConnection的封裝.
AFHTTPSessionManager 對NSURLSession的封裝.
1.創建管理者
AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];
2.設置管理者的數據解析類型,默認為 json 格式的解析,可手動修改為其他類型,如 XML:
manager.responseSerializer = [AFXMLParserResponseSerializer serializer];
3.發送網絡請求
NSDictionary *dict = @{@"username":@"zhangsan",@"password":@"zhang"}; [[AFHTTPSessionManager manager] POST:@"http://localhost/login/login.php" parameters:dict progress:^(NSProgress * _Nonnull uploadProgress) { // } success:^(NSURLSessionDataTask * _Nonnull task, id _Nullable responseObject) { // NSLog(@"responseObject:%@",responseObject); } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) { // NSLog(@"error:%@",error); }]; [[AFHTTPSessionManager manager] GET:@"http://localhost/login/login.php" parameters:dict progress:^(NSProgress * _Nonnull downloadProgress) { NSLog(@"下載進度:%@",downloadProgress); } success:^(NSURLSessionDataTask * _Nonnull task, id _Nullable responseObject) { NSLog(@"responseObject:%@",responseObject); } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) { NSLog(@"error:%@",error); }];
3.1 首先要明確發送的是什麼類型的請求(GET/POST/HEAD...)
3.2 AFN 3.0之後的網絡接口相比之前的網絡接口多了一個參數:網絡進度.
參數:
1. urlString: 網絡接口地址.
2. parameters: 參數字典.key:服務器接收普通參數的key值,value就是參數內容.
3. progress: 網絡進度
4. success: 成功回調
5. failure: 失敗回調
3.3 AFN根據 response.MIMEType 來判斷服務器返回數據的類型. 如果類型不匹配,但是又是JSON數據,解決方案:
1.改變解析器類型為:萬能解析器 ---> 手動解析JSON
manager.responseSerializer = [AFHTTPResponseSerializer serializer];
2.改變判斷條件的類型,以使類型匹配,acceptableContentTypes默認情況下無 text/plain 類型
manager.responseSerializer.acceptableContentTypes = [NSSet setWithObjects:@"application/json", @"text/json", @"text/javascript",@"text/plain",nil];
注: 如果沒有使用第三方框架(CocoaPods)來管理第三方框架,可以直接修改第三方框架的源代碼.
一般在開發中,不要隨意修改第三方源碼.
1.創建管理者
AFHTTPRequestOperationManager *mgr = [AFHTTPRequestOperationManager manager];
2.封裝請求參數
NSMutableDictionary *params = [NSMutableDictionary dictionary]; params[@"username"] = @"哈哈哈"; params[@"pwd"] = @"123";
3.發送請求
NSString *url = @"http://localhost:8080/Server/login"; [mgr POST:url parameters:params success:^(AFHTTPRequestOperation *operation, id responseObject) { // 請求成功的時候調用這個block NSLog(@"請求成功---%@", responseObject); } failure:^(AFHTTPRequestOperation *operation, NSError *error) { // 請求失敗的時候調用調用這個block NSLog(@"請求失敗"); }]; // GET請求 [mgr GET:url parameters:params success:^(AFHTTPRequestOperation *operation, id responseObject) { // 請求成功的時候調用這個block NSLog(@"請求成功---%@", responseObject); } failure:^(AFHTTPRequestOperation *operation, NSError *error) { // 請求失敗的時候調用調用這個block NSLog(@"請求失敗"); }];
1.AFN可以自動對服務器返回的數據進行解析
* 默認將服務器返回的數據當做JSON來解析
2.設置對服務器返回數據的解析方式
1> 當做是JSON來解析(默認做法)
* mgr.responseSerializer = [AFJSONResponseSerializer serializer];
* responseObject的類型是NSDictionary或者NSArray
2> 當做是XML來解析
* mgr.responseSerializer = [AFXMLParserResponseSerializer serializer];
* responseObject的類型是NSXMLParser
3> 直接返回data
* 意思是:告訴AFN不要去解析服務器返回的數據,保持原來的data即可
* mgr.responseSerializer = [AFHTTPResponseSerializer serializer];
3.注意
* 服務器返回的數據一定要跟responseSerializer對得上
1> 服務器返回的是JSON數據
* AFJSONResponseSerializer
* AFHTTPResponseSerializer
2> 服務器返回的是XML數據
* AFXMLParserResponseSerializer
* AFHTTPResponseSerializer
3> 服務器返回的是其他數據
* AFHTTPResponseSerializer
可利用 AFN 實時監測網絡狀態.
AFNetworkReachabilityManager 實時檢測網絡狀態改變的類 AFNetworkReachabilityManager *manager = [AFNetworkReachabilityManager sharedManager]; // 設置網絡狀態改變之後的操作 [manager setReachabilityStatusChangeBlock:^(AFNetworkReachabilityStatus status) { // status :當前的網絡狀態. // AFNetworkReachabilityStatusUnknown // AFNetworkReachabilityStatusNotReachable // AFNetworkReachabilityStatusReachableViaWWAN // AFNetworkReachabilityStatusReachableViaWiFi switch (status) { case AFNetworkReachabilityStatusUnknown: NSLog(@"未知的網絡狀態"); break; case AFNetworkReachabilityStatusNotReachable: NSLog(@"沒有網絡"); break; case AFNetworkReachabilityStatusReachableViaWWAN: NSLog(@"蜂窩移動網絡"); break; case AFNetworkReachabilityStatusReachableViaWiFi: NSLog(@"WIFI網絡"); break; default: break; } }]; // 開始檢測網絡狀態 [manager startMonitoring];
HTTPS = HTTP(超文本傳輸協議) + SSL (安全連接層) HTTP 的安全版本.
HTTPS 會專門建立一個 安全的數據傳輸通道來傳輸數據,外界拿不到任何數據,現階段最安全的協議,目前在 http 模式下三大運營商發送的惡意廣告泛濫,並且可以獲得用戶的個人信息,知乎有專門文章講解如何到工信部投訴的內容。
HTTPS 需要數字驗證,目前很多大公司使用的數字驗證都是默認支持的.
[manager GET:@"https://mail.itcast.cn" parameters:nil progress:^(NSProgress * _Nonnull downloadProgress) { // } success:^(NSURLSessionDataTask * _Nonnull task, id _Nullable responseObject) { NSData *data = responseObject; // NSLog(@"成功:%@",[[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding]); } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) { // NSLog(@"失敗:%@",error); }];
AFN 默認情況下就是支持 HTTPS 訪問的,但是如果用 HTTPS 的方式訪問未受信任的網站便會報錯,解決方案:
修改對 SSL 的檢測:
AFN3.0之前:
manager.securityPolicy.allowInvalidCertificates = YES;
AFN3.0之後:
manager.securityPolicy.validatesDomainName = NO;
一、底層實現
1> AFN的底層基於OC的NSURLConnection和NSURLSession
2> ASI的底層基於純C語言的CFNetwork框架
3> ASI的運行性能 高於 AFN
二、對服務器返回的數據處理
1> ASI沒有直接提供對服務器數據處理的方式,直接返回data\string
2> AFN提供了多種對服務器數據處理的方式
* JSON處理
* XML處理
* 其他處理
三、監聽請求的過程
1> AFN提供了success和failure兩個block來監聽請求的過程(只能監聽成功和失敗)(3.0以後添加了進度)
* success : 請求成功後調用
* failure : 請求失敗後調用
2> ASI提供了3套方案,每一套方案都能監聽請求的完整過程
(監聽請求開始、接收到響應頭信息、接受到具體數據、接受完畢、請求失敗)
* 成為代理,遵守協議,實現協議中的代理方法
* 成為代理,不遵守協議,自定義代理方法
* 設置block
四、在文件下載和文件上傳的使用難易度
1> AFN
* 不容易監聽下載進度和上傳進度
* 不容易實現斷點續傳
* 一般只用來下載不大的文件
2> ASI
* 非常容易實現下載和上傳
* 非常容易監聽下載進度和上傳進度
* 非常容易實現斷點續傳
* 下載或大或小的文件都行
五、ASI提供了更多的實用功能
1> 控制圈圈要不要在請求過程中轉
2> 可以輕松地設置請求之間的依賴:每一個請求都是一個NSOperation對象
3> 可以統一管理所有請求(還專門提供了一個叫做ASINetworkQueue來管理所有的請求對象)
* 暫停\恢復\取消所有的請求
* 監聽整個隊列中所有請求的下載進度和上傳進度