iOS ASIHTTPRequest詳解

ASIHTTPRequest對CFNetwork API進行了封裝,而且使用起來很是簡單,用Objective-C編寫,能夠很好的應用在Mac OS X系統和iOS平臺的應用程序中。ASIHTTPRequest適用於基本的HTTP請求,和基於REST的服務之間的交互。 緩存

ASIHTTPRequest功能很強大,主要特點以下: 安全

  • l 經過簡單的接口,便可完成向服務端提交數據和從服務端獲取數據的工做
  • l 下載的數據,可存儲到內存中或直接存儲到磁盤中
  • l 能上傳本地文件到服務端
  • l 能夠方便的訪問和操做請求和返回的Http頭信息
  • l 能夠獲取到上傳或下載的進度信息,爲應用程序提供更好的體驗
  • l 支持上傳或下載隊列,而且可獲取隊列的進度信息
  • l 支持基本、摘要和NTLM身份認證,在同一會話中受權憑證會自動維持,而且能夠存儲在Keychain(Mac和iOS操做系統的密碼管理系統)中
  • l 支持Cookie
  • l 當應用(iOS 4+)在後臺運行時,請求能夠繼續運行
  • l 支持GZIP壓縮數據
  • l 內置的ASIDownloadCache類,能夠緩存請求返回的數據,這樣即便沒有網絡也能夠返回已經緩存的數據結果
  • l ASIWebPageRequest –能夠下載完整的網頁,包括包含的網頁、樣式表、腳本等資源文件,並顯示在UIWebView /WebView中。任意大小的頁面均可以無限期緩存,這樣即便沒有網絡也能夠離線瀏覽
  • l 支持客戶端證書
  • l 支持經過代理髮起Http請求
  • l 支持帶寬限制。在iOS平臺,能夠根據當前網絡狀況來自動決定是否限制帶寬,例如當使用WWAN(GPRS/Edge/3G)網絡時限制,而當使用WIFI時不作任何限制
  • l 支持斷點續傳
  • l 支持同步和異步請求


1) 添加文件

往一個Xcode項目中添加第三方類庫文件,有兩種方式: 服務器

1. 第一種方式,在Finder中打開須要添加到文件或文件夾,在Xcode中打開要添加文件的項目,而後選中要添加的文件或文件夾,將它從Finder中拖到Xcode中,而後釋放。在彈出的對話框中,若是文件已經拷貝到了項目文件目錄中,則不須要選中「Copy items」的複選框;若是文件沒有拷貝到項目文件目錄,就須要選中「Copy items」的複選框,這樣Xcode會自動把文件複製到項目文件目錄下。以下圖所示:
clip_image002
clip_image004 cookie

2. 第二種方式,在Xcode中,在要添加文件的分組下點右鍵,選中「Add Files to 「My Project」…」菜單,在彈出的文件瀏覽對話框中選中要添加到文件或文件夾。若是要添加文件已經拷貝到了項目文件目錄中,則不須要選中「Copy items」的複選框;若是文件沒有拷貝到項目文件目錄,就須要選中「Copy items」的複選框,這樣Xcode會自動把文件複製到項目文件目錄下。以下圖所示:
clip_image006
clip_image008 網絡

根據上面的說明,添加ASIHTTPRequest相關文件到Xcode項目中,所需文件列表以下: session

ASIHTTPRequestConfig.h 架構

ASIHTTPRequestDelegate.h app

ASIProgressDelegate.h 異步

ASICacheDelegate.h fetch

ASIHTTPRequest.h

ASIHTTPRequest.m

ASIDataCompressor.h

ASIDataCompressor.m

ASIDataDecompressor.h

ASIDataDecompressor.m

ASIFormDataRequest.h

ASIInputStream.h

ASIInputStream.m

ASIFormDataRequest.m

ASINetworkQueue.h

ASINetworkQueue.m

ASIDownloadCache.h

ASIDownloadCache.m

ASIAuthenticationDialog.h

ASIAuthenticationDialog.m

Reachability.h (在源碼的 External/Reachability 目錄下)

Reachability.m (在源碼的 External/Reachability 目錄下)

2) 連接相關類庫

1. 選中項目

2. 選中目標

3. 跳轉到「Build Phases」標籤

4. 展開「Link Binary With Libraries」分組

5. 點擊「+」添加類庫

以下圖所示:

clip_image010

6. 從列表中選擇CFNetwork.framework,而後點擊「Add」按鈕。

clip_image012

7. 按照上一步相同的方法添加:SystemConfiguration.framework, MobileCoreServices.framework,CoreGraphics.framework和libz.1.2.3.dylib這幾個類庫。

8. 添加完後,能夠將添加好的一塊兒類庫拖到Xcode項目的Frameworks目錄下
clip_image014



在使用ASIHTTPRequest以前,請確認已經正確安裝,而後在須要應用它的代碼文件頭部,加入:

#import 「ASIHTTPRequest.h」

這樣就能夠在代碼中使用ASIHTTPRequest相關的類。

發起一個同步請求

同步意爲着線程阻塞,在主線程中使用此方法會使應用Hang住而不響應任何用戶事件。因此,在應用程序設計時,大多被用在專門的子線程增長用戶體驗,或用異步請求代替(下面會講到)。

- (IBAction)grabURL:(id)sender
{
  NSURL *url = [NSURL URLWithString:@" http://allseeing-i.com"];
  ASIHTTPRequest *request = [ASIHTTPRequest requestWithURL:url];
  [request startSynchronous];
  NSError *error = [request error];
  if(!error) {
    NSString *response = [request responseString];
  }
}

a, 用requestWithURL快捷方法獲取ASIHTTPRequest的一個實例
b, startSynchronous 方法啓動同步訪問,
c, 因爲是同步請求,沒有基於事件的回調方法,因此從request的error屬性獲取錯誤信息。
d, responseString,爲請求的返回NSString信息。

建立一個異步請求

異步請求的好處是不阻塞當前線程,但相對於同步請求略爲複雜,至少要添加兩個回調方法來獲取異步事件。
下面異步請求代碼完成上面一樣的一件事情:

- (IBAction)grabURLInBackground:(id)sender
{
   NSURL *url = [NSURL URLWithString:@" http://allseeing-i.com"];
   ASIHTTPRequest *request = [ASIHTTPRequest requestWithURL:url];
   [request setDelegate:self];
   [request startAsynchronous];
}
 
- (void)requestFinished:(ASIHTTPRequest *)request
{
   // Use when fetching text data
   NSString *responseString = [request responseString];
 
   // Use when fetching binary data
   NSData *responseData = [request responseData];
}
 
- (void)requestFailed:(ASIHTTPRequest *)request
{
   NSError *error = [request error];
}

a,與上面不一樣的地方是指定了一個 "delegate",並用startAsynchronous來啓動網絡請求。
b,在這裏實現了兩個delegate的方法,當數據請求成功時會調用requestFinished,請求失敗時(如網絡問題或服務器內部錯誤)會調用requestFailed。

隊列請求

提供了一個對異步請求更加精準豐富的控制。
如,能夠設置在隊列中,同步請求的鏈接數。往隊列裏添加的請求實例數大於maxConcurrentOperationCount時,請求實例將被置爲等待,直到前面至少有一個請求完成並出列才被放到隊列裏執行。
也適用於當咱們有多個請求需求按順序執行的時候(多是業務上的須要,也多是軟件上的調優),僅僅須要把maxConcurrentOperationCount設爲「1」。

- (IBAction)grabURLInTheBackground:(id)sender
{
   if(![self queue]) {
      [self setQueue:[[[NSOperationQueue alloc] init] autorelease]];
   }
 
   NSURL *url = [NSURL URLWithString:@" http://allseeing-i.com"];
   ASIHTTPRequest *request = [ASIHTTPRequest requestWithURL:url];
   [request setDelegate:self];
   [request setDidFinishSelector:@selector(requestDone:)];
   [request setDidFailSelector:@selector(requestWentWrong:)];
   [[self queue] addOperation:request];//queue is an NSOperationQueue
}
 
- (void)requestDone:(ASIHTTPRequest *)request
{
   NSString *response = [request responseString];
}
 
- (void)requestWentWrong:(ASIHTTPRequest *)request
{
   NSError *error = [request error];
}

建立NSOperationQueue,這個Cocoa架構的執行任務(NSOperation)的任務隊列。咱們經過ASIHTTPRequest.h的源碼能夠看到,此類自己就是一個NSOperation的子類。也就是說它能夠直接被放到"任務隊列"中,並被執行。上面的代碼隊了隊列的建立與添加操做外,其它代碼與上一例同樣。

請求隊列上下文

a,能夠設置一個上下文(userInfo)到request對象中,當請求響應完後能夠經過訪問request對象的userInfo獲取裏面的信息
b,爲每個請求實例設置不一樣的setDidFinishSelector / setDidFailSelector的回調方法
c,子類化ASIHTTPRequest,重寫requestFinished: 與 failWithProblem:方法

ASINetworkQueues, 它的delegate提供更爲豐富的功能

提供的更多的回調方法以下:
a,requestDidStartSelector,請求發起時會調此方法,你能夠在此方法中跟據業務選擇性的設置request對象的deleaget。
b,requestDidReceiveResponseHeadersSelector,當接受完響應的Header後設計此方法,這個對下載大數據的時候至關有用,你能夠在方法裏作更多業務上的處理。
c,requestDidFinishSelector,請求並響應成功完成時調用此方法
d,requestDidFailSelector,請求失敗
e,queueDidFinishSelector,整個隊列裏的全部請求都結束時調用此方法。

它是NSOperationQueues的擴展,小而強大。但也與它的父類略有區別。如,僅添加到隊列中其實並不能執行請求,只有調用[ queue g o]纔會執行;一個正在運行中的隊列,並不須要重複調用[ queue go ]。

默認狀況下,隊列中的一個請求若是失敗,它會取消全部未完成的請求。能夠設置[ queue setShouldCancelAllRequestsOnFailure:NO ]來修 正。

取消異步請求

首先,同步請求是不能取消的。
其次,不論是隊列請求,仍是簡單的異步請求,所有調用[ request cancel ]來取消請求。

取消的請求默認都會按請求失敗處理,並調用請求失敗delegate。
若是不想調用delegate方法,則設置:[ request clearDelegatesAndCancel];

隊列請求中須要注意的是,若是你取消了一個請求,隊列會自動取消其它全部請求。
若是隻想取消一個請求,能夠設置隊列:[ queue setShouldCancelAllRequestsOnFailure:NO ];
若是想明確取消全部請求:[ queue cancelAllOperations ];

安全的內存回收建議

request並無retain你的delegate,因此在沒有請求完的時候釋放了此delegate,須要在dealloc方法裏先取消全部請求,再釋放請求實例,如:

- (void)dealloc
{
   [request clearDelegatesAndCancel];
   [request release];
   ...
   [superdealloc];
}
向服務器端上傳數據

ASIFormDataRequest ,模擬 Form表單提交,其提交格式與 Header會自動識別。
沒有文件:application/x-www-form-urlencoded
有文件:multipart/form-data

ASIFormDataRequest *request = [ASIFormDataRequest requestWithURL:url];
[request setPostValue:@"Ben"forKey:@"first_name"];
[request setPostValue:@"Copsey"forKey:@"last_name"];
[request setFile:@"/Users/ben/Desktop/ben.jpg"forKey:@"photo"];
[request addData:imageData withFileName:@"george.jpg"andContentType:@"image/jpeg"forKey:@"photos"];

若是要發送自定義數據:

ASIHTTPRequest *request = [ASIHTTPRequest requestWithURL:url];
[request appendPostData:[@"This is my data"dataUsingEncoding:NSUTF8StringEncoding]];
// Default becomes POST when you use appendPostData: / appendPostDataFromFile: / setPostBody:
[request setRequestMethod:@"PUT"];
下載文件

經過設置request的setDownloadDestinationPath,能夠設置下載文件用的下載目標目錄。
首先,下載過程文件會保存在temporaryFileDownloadPath目錄下。若是下載完成會作如下事情:
1,若是數據是壓縮的,進行解壓,並把文件放在downloadDestinationPath目錄中,臨時文件被刪除
2,若是下載失敗,臨時文件被直接移到downloadDestinationPath目錄,並替換同名文件。

若是你想獲取下載中的全部數據,能夠實現delegate中的request:didReceiveData:方法。但若是你實現了這個方法,request在下載完後,request並不把文件放在downloadDestinationPath中,須要手工處理。

獲取響應信息

信息:status , header, responseEncoding

[request responseStatusCode];
[[request responseHeaders] objectForKey:@"X-Powered-By"];
 [request responseEncoding];
獲取請求進度

有兩個回調方法能夠獲取請求進度,
1,downloadProgressDelegate,能夠獲取下載進度
2,uploadProgressDelegate,能夠獲取上傳進度

cookie的支持

若是Cookie存在的話,會把這些信息放在NSHTTPCookieStorage容器中共享,並供下次使用。
你能夠用[ ASIHTTPRequest setSessionCookies:nil ] ; 清空全部Cookies。
固然,你也能夠取消默認的Cookie策略,而使自定義的Cookie:

//Create a cookie
NSDictionary *properties = [[[NSMutableDictionary alloc] init] autorelease];
[properties setValue:[@"Test Value"encodedCookieValue] forKey:NSHTTPCookieValue];
[properties setValue:@"ASIHTTPRequestTestCookie"forKey:NSHTTPCookieName];
[properties setValue:@".allseeing-i.com"forKey:NSHTTPCookieDomain];
[properties setValue:[NSDate dateWithTimeIntervalSinceNow:60*60] forKey:NSHTTPCookieExpires];
[properties setValue:@"/asi-http-request/tests"forKey:NSHTTPCookiePath];
NSHTTPCookie *cookie = [[[NSHTTPCookie alloc] initWithProperties:properties] autorelease];
 
//This url will return the value of the 'ASIHTTPRequestTestCookie' cookie
request = [ASIHTTPRequest requestWithURL:url];
[request setUseCookiePersistence:NO];
[request setRequestCookies:[NSMutableArray arrayWithObject:cookie]];
[request startSynchronous];
 
//Should be: I have 'Test Value' as the value of 'ASIHTTPRequestTestCookie'
NSLog(@"%@",[request responseString]);
大文件斷點續傳

0.94之後支持大文件的斷點下載,只須要設置:
[ request setAllowResumeForFileDownloads:YES ];
[ request setDownloadDestinationPath:downloadPath ];
就能夠了。

ASIHTTPRequest會自動保存訪問過的URL信息,並備以後用。在如下幾個場景很是有用:
1,當沒有網絡鏈接的時候。
2,已下載的數據再次請求時,僅當它與本地版本不樣時才進行下載。

ASIDownloadCache 設置下載緩存

它對Get請求的響應數據進行緩存(被緩存的數據必需是成功的200請求):

[ASIHTTPRequest setDefaultCache:[ASIDownloadCache sharedCache]];

當設置緩存策略後,全部的請求都被自動的緩存起來。
另外,若是僅僅但願某次請求使用緩存操做,也能夠這樣使用:

ASIHTTPRequest *request = [ASIHTTPRequest requestWithURL:url];
[request setDownloadCache:[ASIDownloadCache sharedCache]];
多種的緩存並存

僅僅須要建立不一樣的ASIDownloadCache,並設置緩存所使用的路徑,並設置到須要使用的request實例中:

ASIDownloadCache *cache = [[[ASIDownloadCache alloc] init] autorelease];
[cache setStoragePath:@"/Users/ben/Documents/Cached-Downloads"];
[self setMyCache:cache];
ASIHTTPRequest *request = [ASIHTTPRequest requestWithURL:url];
[request setDownloadCache:[self myCache]];
緩存策略

緩存策略是咱們控制緩存行爲的主要方式,如:何時進行緩存,緩存數據的利用方式。
如下是策略可選列表(可組合使用):

ASIUseDefaultCachePolicy 這是一個默認的緩存策略「ASIAskServerIfModifiedWhenStaleCachePolicy」,這個很明白,見名知意(它不能與其它策略組合使用)
ASIDoNotReadFromCacheCachePolicy 所讀數據不使用緩存
ASIDoNotWriteToCacheCachePolicy 不對緩存數據進行寫操做
ASIAskServerIfModifiedWhenStaleCachePolicy 默認緩存行爲,request會先判斷是否存在緩存數據。a, 若是沒有再進行網絡請求。 b,若是存在緩存數據,而且數據沒有過時,則使用緩存。c,若是存在緩存數據,但已通過期,request會先進行網絡請求,判斷服務器版本與本地版本是否同樣,若是同樣,則使用緩存。若是服務器有新版本,會進行網絡請求,並更新本地緩存
ASIAskServerIfModifiedCachePolicy 與默認緩存大體同樣,區別僅是每次請求都會 去服務器判斷是否有更新
ASIOnlyLoadIfNotCachedCachePolicy 若是有緩存在本地,無論其過時與否,總會拿來使用
ASIDontLoadCachePolicy 僅當有緩存的時候纔會被正確執行,若是沒有緩存,request將被取消(沒有錯誤信息)
ASIFallbackToCacheIfLoadFailsCachePolicy 這個選項常常被用來與其它選項組合使用。請求失敗時,若是有緩存當網絡則返回本地緩存信息(這個在處理異常時很是有用)
若是設置了「defaultCachePolicy」則全部的請求都會使用此緩存。
緩存存儲方式

你能夠設置緩存的數據須要保存多長時間,ASIHTTPRequest提供了兩種策略:
a,ASICacheForSessionDurationCacheStoragePolicy,默認策略,基於session的緩存數據存儲。當下次運行或[ASIHTTPRequest clearSession]時,緩存將失效。
b,ASICachePermanentlyCacheStoragePolicy,把緩存數據永久保存在本地,
如:

ASIHTTPRequest *request = [ ASIHTTPRequest requestWithURL:url ];
[ request setCacheStoragePolicy:ASICachePermanentlyCacheStoragePolicy ];

另外,也可使用clearCachedResponsesForStoragePolicy來清空指定策略下的緩存數據。

緩存其它特性

設置是否按服務器在Header裏指定的是否可被緩存或過時策略進行緩存:

[[ ASIDownloadCache sharedCache ] setShouldRespectCacheControlHeaders:NO ];

設置request緩存的有效時間:

[ request setSecondsToCache:60*60*24*30];// 緩存30天

能夠判斷數據是否從緩存讀取:

[ request didUseCachedResponse ];

設置緩存所使用的路徑:

[ request setDownloadDestinationPath:[[ ASIDownloadCache sharedCache ] pathToStoreCachedResponseDataForRequest:request ]];
實現自定義的緩存

只要簡單的實現ASICacheDelegate接口就能夠被用來使用。

使用代理請求

默認的狀況下,ASIHTTPRequest會使用被設置的默認代理。但你也能夠手動修改http代理:

// Configure a proxy server manually
NSURL *url = [ NSURL URLWithString:@" http://allseeing-i.com/ignore"];
ASIHTTPRequest *request = [ ASIHTTPRequest requestWithURL:url ];
[ request setProxyHost:@"192.168.0.1"];
[ request setProxyPort:3128];
 
// Alternatively, you can use a manually-specified Proxy Auto Config file (PAC)
// (It's probably best if you use a local file)
[request setPACurl:[NSURL URLWithString:@"file:///Users/ben/Desktop/test.pac"]];
ASIHTTPRequest, 請求的其它特性

iOS4中,當應用後臺運行時仍然請求數據:

[ request setShouldContinueWhenAppEntersBackground:YES ];

是否有網絡請求:

[ ASIHTTPRequest isNetworkInUse ]

是否顯示網絡請求信息在status bar上:

[ ASIHTTPRequest setShouldUpdateNetworkActivityIndicator:NO ];

設置請求超時時,設置重試的次數:

[ request setNumberOfTimesToRetryOnTimeout:2];

KeepAlive的支持:

// Set the amount of time to hang on to a persistent connection before it should expire to 2 minutes
[ request setPersistentConnectionTimeoutSeconds:120];
 
// Disable persistent connections entirely
[ request setShouldAttemptPersistentConnection:NO ];
相關文章
相關標籤/搜索