iOS開發ASIHTTPRequest進度追蹤
本文為大家介紹了發ASIHTTPRequest進度追蹤的內容,其中包括追蹤單個request的下載進度,追蹤一系列request的下載進度,追蹤單個request的上傳進度,追蹤一系列request的上傳進度,精確進度條vs簡單進度條,自定義進度追蹤等等內容。
每個ASIHTTPRequest有兩個delegate用來追蹤進度:
downloadProgressDelegate (下載) 和 uploadProgressDelegate (上載)。
進度delegate可以是NSProgressIndicators (Mac OS X) 或者 UIProgressViews (iPhone).ASIHTTPRequest會自適應這兩個class的行為。你也可以使用自定義class作為進度delegate,只要它響應setProgress:函數。
- 如果你執行單個request,那么你需要為該request設定upload/download進度delegate
- 如果你在進行多個請求,并且你想要追蹤整個隊列中的進度,你必須使用ASINetworkQueue并設置隊列的進度delegate
- 如果上述兩者你想同時擁有,恭喜你,0.97版以后的ASIHTTPRequest,這個可以有 ^ ^
IMPORTANT:如果你向一個要求身份驗證的網站上傳數據,那么每次授權失敗,上傳進度條就會被重置為上一次的進度值。因此,當與需要授權的web服務器交互時,建議僅當useSessionPersistence為YES時才使用上傳進度條,并且確保你在追蹤大量數據的上傳進度之前,先使用另外的request來進行授權。
追蹤小于128KB的數據上傳進度目前無法做到,而對于大于128kb的數據,進度delegate不會收到***個128kb數據塊的進度信息。這是因為CFNetwork庫API的限制。我們曾向apple提交過bug報告(bug id 6596016),希望apple能修改CFNetwork庫以便實現上述功能。
2009-6-21:Apple的哥們兒們真棒!iPhone 3.0 SDK里,buffer大小已經被減小到32KB了,我們的上傳進度條可以更精確了。
追蹤單個request的下載進度
這個例子中, myProgressIndicator是個 NSProgressIndicator.
- ASIHTTPRequest *request = [ASIHTTPRequest requestWithURL:url];
- [request setDownloadProgressDelegate:myProgressIndicator];
- [request startSynchronous];
- NSLog(@"Max: %f, Value: %f", [myProgressIndicator maxValue],[myProgressIndicator doubleValue]);
追蹤一系列request的下載進度
在這個例子中, myProgressIndicator 是個 UIProgressView, myQueue是個 ASINetworkQueue.
- - (void)fetchThisURLFiveTimes:(NSURL *)url
- {
- [myQueue cancelAllOperations];
- [myQueue setDownloadProgressDelegate:myProgressIndicator];
- [myQueue setDelegate:self];
- [myQueue setRequestDidFinishSelector:@selector(queueComplete:)];
- int i;
- for (i=0; i<5; i++) {
- ASIHTTPRequest *request = [ASIHTTPRequest requestWithURL:url];
- [myQueue addOperation:request];
- }
- [myQueue go];
- }
- - (void)queueComplete:(ASINetworkQueue *)queue
- {
- NSLog(@"Value: %f", [myProgressIndicator progress]);
- }
這個例子中,我們已經為ASINetworkQueues調用過[myQueue go]了。
追蹤單個request的上傳進度
在這個例子中, myProgressIndicator 是個 UIProgressView。
- ASIFormDataRequest *request = [ASIFormDataRequest requestWithURL:url];
- [request setPostValue:@"Ben" forKey:@"first_name"];
- [request setPostValue:@"Copsey" forKey:@"last_name"];
- [request setUploadProgressDelegate:myProgressIndicator];
- [request startSynchronous];
- NSLog(@"Value: %f",[myProgressIndicator progress]);
追蹤一系列request的上傳進度
這個例子中, myProgressIndicator是個 NSProgressIndicator, myQueue是個ASINetworkQueue.
- - (void)uploadSomethingFiveTimes:(NSURL *)url
- {
- [myQueue cancelAllOperations];
- [myQueue setUploadProgressDelegate:myProgressIndicator];
- [myQueue setDelegate:self];
- [myQueue setRequestDidFinishSelector:@selector(queueComplete:)];
- int i;
- for (i=0; i<5; i++) {
- ASIHTTPRequest *request = [ASIFormDataRequest requestWithURL:url];
- [request setPostBody:[@"Some data" dataUsingEncoding:NSUTF8StringEncoding]];
- [myQueue addOperation:request];
- }
- [myQueue go];
- }
- - (void)queueComplete:(ASINetworkQueue *)queue
- {
- NSLog(@"Max: %f, Value: %f", [myProgressIndicator maxValue],[myProgressIndicator doubleValue]);
- }
精確進度條vs簡單進度條
ASIHTTPRequest提供兩種進度條顯示,簡單進度條和精確進度條,使用ASIHTTPRequests 和ASINetworkQueues的showAccurateProgress 來控制。為一個request設置showAccurateProgress只會對該request有效。如果你為一個隊列設置showAccurateProgress,那么會影響隊列里所有的request。
簡單進度條
當使用簡單進度條時,進度條只會在一個request完成時才更新。對于單個request,這意味著你只有兩個進度狀態:0%和100%。對于一個有5個request的隊列來說,有五個狀態:0%,25%,50%,75%,100%,每個request完成時,進度條增長一次。
簡單進度條(showAccurateProgress = NO)是ASINetworkQueue的默認值,適用于大量小數據請求。
精確進度條
當使用精確進度條時,每當字節被上傳或下載時,進度條都會更新。它適用于上傳/下載大塊數據的請求,并且會更好的顯示已經發送/接收的數據量。
使用精確進度條追蹤上傳會輕微降低界面效率,因為進度delegate(一般是UIProgressViews 或NSProgressIndicators)會更頻繁地重繪。
使用精確進度條追蹤下載會更影響界面效率,因為隊列會先為每個GET型request進行HEAD請求,以便統計總下載量。強烈推薦對下載大文件的隊列使用精確進度條,但是要避免對大量小數據請求使用精確進度條。
精確進度條(showAccurateProgress = YES)是以同步方式執行的ASIHTTPRequest的默認值。
自定義進度追蹤
ASIProgressDelegate 協議定義了所有能更新一個request進度的方法。多數情況下,設置你的uploadProgressDelegate或者 downloadProgressDelegate為NSProgressIndicator或者UIProgressView會很好。但是,如果你想進行更復雜的追蹤,你的進度delegate實現下列函數要比 setProgress: (iOS) 或者 setDoubleValue: / setMaxValue: (Mac)好:
這些函數允許你在實際量的數據被上傳或下載時更新進度,而非簡單方法的0到1之間的數字。
downloadProgressDelegates方法
- request:didReceiveBytes: 每次request下載了更多數據時,這個函數會被調用(注意,這個函數與一般的代理實現的 request:didReceiveData:函數不同)。
- request:incrementDownloadSizeBy: 當下載的大小發生改變時,這個函數會被調用,傳入的參數是你需要增加的大小。這通常發生在request收到響應頭并且找到下載大小時。
uploadProgressDelegates方法
- request:didSendBytes: 每次request可以發送更多數據時,這個函數會被調用。注意:當一個request需要消除上傳進度時(通常是該request發送了一段數據,但是因為授權失敗或者其他什么原因導致這段數據需要重發)這個函數會被傳入一個小于零的數字。