前言
使用 iOS SDK 中的 HTTP 網絡請求 API,相當的復雜,調用很繁瑣,ASIHTTPRequest 就是一個對 CFNetwork API 進行了封裝,並且使用起來非常簡單的一套 API,外號 “HTTP終結者”,用 Objective-C 編寫,運行效率很高,可以很好的應用在 Mac OS X 系統和 iOS 平台的應用程序中,ASIHTTPRequest 適用於基本的 HTTP 請求,和基於 REST 的服務之間的交互。可惜作者早已停止更新,有一些潛在的 BUG 無人去解決,很多公司的舊項目裡面都殘留著它的身影,以前的很多 iOS 項目都是 ASI + SBJson,會不會用 ASI,可以算是檢驗是否為老牌 iOS 程序員的標准之一。從 iOS 9 開始 CFNetwork 相關的類和方法開始被廢棄,可以使用 AFNetworking 替換 ASIHTTPRequest 的使用。在 iOS 9+ 中使用 ASIHTTPRequest 無需對 App Transport Security Settings 添加設置。
1、ASIHTTPRequest
1.1 ASI 主要特色
1.2 AFN 與 ASI 的區別
1、底層實現
1)AFN 的底層實現基於 OC 的 NSURLConnection 和 NSURLSession
2)ASI 的底層實現基於純 C 語言的 CFNetwork 框架
3)因為 NSURLConnection 和 NSURLSession 是在 CFNetwork 之上的一層封裝,因此 ASI 的運行性能高於 AFN
2、對服務器返回的數據處理
1)ASI 沒有直接提供對服務器數據處理的方式,直接返回的是 NSData/NSString
2)AFN 提供了多種對服務器數據處理的方式 (1) JSON 處理-直接返回 NSDictionary 或者 NSArray
(2) XML 處理-返回的是 xml 類型數據,需對其進行解析
(3) 其他類型數據處理
3、監聽請求過程 1
)AFN 提供了success 和 failure 兩個 block 來監聽請求的過程(只能監聽成功和失敗)
2)ASI 提供了 3 套方案,每一套方案都能監聽請求的完整過程(監聽請求開始、接收到響應頭信息、接受到具體數據、接受完畢、請求失敗)
4、在文件下載和文件上傳的使用難易度
1)AFN
2)ASI
3)實現下載上傳推薦使用 ASI
5、網絡監控
1)AFN 自己封裝了網絡監控類,易使用
2)ASI 使用的是 Reachability,因為使用 CocoaPods 下載 ASI 時,會同步下載 Reachability,但 Reachability 作為網絡監控使用較為復雜(相對於 AFN 的網絡監控類來說)
3)推薦使用 AFN 做網絡監控 AFNetworkReachabilityManager
6、ASI 提供的其他實用功能
1)控制信號旁邊的圈圈要不要在請求過程中轉
2)可以輕松地設置請求之間的依賴:每一個請求都是一個 NSOperation 對象
3)可以統一管理所有請求(還專門提供了一個叫做 ASINetworkQueue 來管理所有的請求對象) 暫停/恢復/取消所有的請求
監聽整個隊列中所有請求的下載進度和上傳進度
2、ASIHTTPRequest 的使用
2.1 添加 ASIHTTPRequest
Github 網址: https://github.com/pokeb/asi-http-request
https://allseeing-i.com/ASIHTTPRequest/
ASIHTTPRequest 系統需求:
ASIHTTPRequest Version
Minimum iOS Target
Target Notes
1.8.1 -> 1.8.2
iOS 3.0+
0.2 -> 1.8.0
ASIHTTPRequest 使用 MRC
Objective-C
// 添加系統庫文件 CFNetwork.framework SystemConfiguration.framework MobileCoreServices.framework CoreGraphics.framework libz.1.1.3.tbd libxml2.2.tbd // 添加第三方庫文件 ASIHTTPRequest-1.8.2 // 在 TARGETS -> Builed Settings -> Search Paths -> Header Search Paths 中添加文件路徑 /usr/include/libxml2 // 在 TARGETS -> Build Phases -> Compile Sources -> ...in .../ASIHTTPRequest 後添加 -fno-objc-arc // 包含頭文件 #import "ASIHTTPRequest.h" #import "ASIFormDataRequest.h"
2.2 ASIHTTPRequest 的設置
Objective-C
// 設置請求頭 ASIHTTPRequest *request = [ASIHTTPRequest requestWithURL:[NSURL URLWithString:@"http://api.hudong.com/iphonexml.do?type=focus-c"]]; [request addRequestHeader:@"Referer" value:@"http://www.dreamingwish.com/"]; // 設置應用後台運行時是否仍然請求數據 request.shouldContinueWhenAppEntersBackground = YES; // 設置請求超時時重試的次數 request.numberOfTimesToRetryOnTimeout = 3; // 設置 KeepAlive 支持 // Set the amount of time to hang on to a persistent connection before it should expire to 2 minutes request.persistentConnectionTimeoutSeconds = 120; // Disable persistent connections entirely request.shouldAttemptPersistentConnection = NO; // 設置是否顯示網絡請求信息在 status bar 上 [ASIHTTPRequest setShouldUpdateNetworkActivityIndicator:NO]; // 網絡狀態檢查 BOOL isNetworkInUse = [ASIHTTPRequest isNetworkInUse];
3、ASI 同步 GET 請求
這是 ASIHTTPRequest 最簡單的一種使用模式,發送 startSynchronous 消息後即開始在同一線程中執行 HTTP 請求,線程將一直等待直到請求結束(請求成功或者失敗)。通過檢查 error 屬性可以判斷請求是否成功或者有錯誤發生。
要獲取返回的文本信息,調用 responseString 方法。如果下載的是二進制文件,例如圖片、MP3,則調用 responseData 方法,可以得到一個 NSData 對象。
一般情況下,應該優先使用異步請求代替同步請求,當在主線程中使用 ASIHTTPRequest 同步請求會阻塞主線程的執行,這導致用戶界面不響應用戶操作,任何動畫都會停止渲染,直到請求完成。
Objective-C
數據請求
NSURL *url = [NSURL URLWithString:@"http://api.hudong.com/iphonexml.do?type=focus-c"]; // 創建請求 ASIHTTPRequest *request = [ASIHTTPRequest requestWithURL:url]; // 設置超時時間,可不設置,使用默認 request.timeOutSeconds = 5; // 發送同步請求 [request startSynchronous]; // 獲得錯誤信息 NSError *error = [request error]; // 網絡請求失敗 if (error) { // 網絡請求成功 NSLog(@"網絡請求失敗:\n%@", error); } else { // 獲得服務器的響應,字符串格式 NSString *responseString = [request responseString]; NSLog(@"網絡請求成功:\n%@", responseString); // 獲得服務器的響應,NSData 格式 NSData *responseData = [request responseData]; textView.text = [[NSString alloc] initWithData:responseData encoding:NSUTF8StringEncoding]; }
文件下載
通過設置 request 的 setDownloadDestinationPath,可以設置下載文件用的下載目標目錄。首先,下載過程文件會保存在 temporaryFileDownloadPath 目錄下。如果下載完成會做以下事情:
1,如果數據是壓縮的,進行解壓,並把文件放在 downloadDestinationPath 目錄中,臨時文件被刪除。
2,如果下載失敗,臨時文件被直接移到 downloadDestinationPath 目錄,並替換同名文件。
如果你想獲取下載中的所有數據,可以實現 delegate 中的 request:didReceiveData:方法。但如果你實現了這個方法,request 在下載完後,request 並不把文件放在 downloadDestinationPath中,需要手工處理。
NSURL *url = [NSURL URLWithString:@"http://www.dreamingwish.com/wp-content/uploads/2011/10/asihttprequest-auth.png"]; ASIHTTPRequest *request = [ASIHTTPRequest requestWithURL:url]; // 設置文件存儲路徑 [request setDownloadDestinationPath:@"/Users/JHQ0228/Desktop/asi.png"]; [request startSynchronous]; // 獲得錯誤信息 NSError *error = [request error]; // 網絡請求失敗 if (error) { NSLog(@"網絡請求失敗:\n%@", error); } else { // 網絡請求成功 NSLog(@"網絡請求成功:\n"); }
4、ASI 異步 GET 請求
請求在後台線程中運行,當請求執行完後再通知調用的線程。這樣不會導致主線程進行網絡請求時,界面被鎖定等情況。
1、協議方式
在這裡實現了兩個 delegate 的方法,當數據請求成功時會調用 requestFinished,請求失敗時(如網絡問題或服務器內部錯誤)會調用 requestFailed。
NSURL *url = [NSURL URLWithString:@"http://api.hudong.com/iphonexml.do?type=focus-c"]; // 創建請求 ASIHTTPRequest *request = [ASIHTTPRequest requestWithURL:url]; // 設置超時時間,可不設置,使用默認 request.timeOutSeconds = 5; // 設置代理,需遵守 <ASIHTTPRequestDelegate> 協議 request.delegate = self; // 發送異步請求 [request startAsynchronous]; // 網絡請求成功,協議方法 - (void)requestFinished:(ASIHTTPRequest *)request { } // 網絡請求失敗,協議方法 - (void)requestFailed:(ASIHTTPRequest *)request { }
2、Block 方式
在平台支持情況下,ASIHTTPRequest 1.8 以上支持 block。
NSURL *url = [NSURL URLWithString:@"http://api.hudong.com/iphonexml.do?type=focus-c"]; // 創建請求,加 __weak 除去 block 循環調用警告 __weak ASIHTTPRequest *request = [ASIHTTPRequest requestWithURL:url]; // 設置超時時間,可不設置,使用默認 request.timeOutSeconds = 5; // 發送異步請求 [request startAsynchronous]; // 網絡請求成功 [request setCompletionBlock:^{ }]; // 網絡請求失敗 [request setFailedBlock:^{ }];
5、ASI POST 請求
1、POST 表單
ASIFormDataRequest,模擬 Form 表單提交,其提交格式與 Header 會自動識別。文件中的數據是需要時才從磁盤加載,所以只要 web server 能處理,那麼上傳大文件是沒有問題的。
// 通常數據是以 'application/x-www-form-urlencoded' 格式發送的,如果上傳了二進制數據或者文件,那麼格式將自動變為 ‘multipart/form-data'。 ASIFormDataRequest *request = [ASIFormDataRequest requestWithURL:[NSURL URLWithString:@"http://www.dreamingwish.com"]]; // 沒有文件 [request setPostValue:@"Ben" forKey:@"first_name"]; [request setPostValue:@"Copsey" forKey:@"last_name"]; // 發送文件 [request setFile:@"/Users/ben/Desktop/ben.jpg" forKey:@"photo"]; // 數據的 mime 頭是自動判定的,但是如果你想自定義mime頭,那麼這樣: ASIFormDataRequest *request = [ASIFormDataRequest requestWithURL:[NSURL URLWithString:@"http://www.dreamingwish.com"]]; // Upload a file on disk [request setFile:@"/Users/ben/Desktop/ben.jpg" withFileName:@"myphoto.jpg" andContentType:@"image/jpeg" forKey:@"photo"]; // Upload an NSData instance NSData *imageData = UIImagePNGRepresentation([UIImage imageNamed:@"myphoto.jpg"]); [request setData:imageData withFileName:@"myphoto.jpg" andContentType:@"image/jpeg" forKey:@"photo"]; // 你可以使用 addPostValue 方法來發送相同 name 的多個數據: ASIFormDataRequest *request = [ASIFormDataRequest requestWithURL:[NSURL URLWithString:@"http://www.dreamingwish.com"]]; [request addPostValue:@"Ben" forKey:@"names"]; [request addPostValue:@"George" forKey:@"names"];
2、PUT 請求、自定義 POST 請求
如果你想發送 PUT 請求,或者你想自定義 POST 請求,使用 appendPostData: 或者 appendPostDataFromFile:
ASIHTTPRequest *request = [ASIHTTPRequest requestWithURL:[NSURL URLWithString:@"http://www.dreamingwish.com"]]; [request appendPostData:[@"This is my data" dataUsingEncoding:NSUTF8StringEncoding]]; // Default becomes POST when you use appendPostData: / appendPostDataFromFile: / setPostBody: [request setRequestMethod:@"PUT"];
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持本站。