程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> C語言 >> C++ >> C++入門知識 >> NSURLConnection同步和異步連接

NSURLConnection同步和異步連接

編輯:C++入門知識

NSURLConnection同步和異步連接


NSURLConnection去加載一個URL請求時候有兩種方式,一種是同步加載,一種是異步加載。

同步加載會阻塞當前的那個線程,如果將同步加載的代碼放在主線程裡去執行,那麼就會阻塞主線程。

異步加載一種方式使用的是block,就算將加載的代碼放到主線程去執行,也不會阻塞主線程。異步加載的另一種方式比較靈活。它可以在你需要的時候去啟動,在你不需要的時候可以取消。

\

先看一個在主線程種同步和異步,以及使用GCD同步三種方式加載數據的阻塞情況

//
//  AppDelegate.m
//  NSURLConnectionDemo
//
//  Created by Lion Mac OS on 5/30/14.
//  Copyright (c) 2014 Yi Gui Hua. All rights reserved.
//

#import "AppDelegate.h"

@implementation AppDelegate

- (void)dealloc
{
    [super dealloc];
}

- (void)applicationDidFinishLaunching:(NSNotification *)aNotification
{
    // Insert code here to initialize your application
    [self fetchYahooData];
    
    NSLog(@"-------------------------------------");
    [self fetchYahooData2_GCD];
     NSLog(@"-------------------------------------");
    [self fetchYahooData3];
}
//使用異步加載,那麼就不會阻塞主線程,因為異步他會開啟一個子線程去加載


//不使用GCD,同步加在數據,會阻塞主線程
-(void)fetchYahooData{
    NSLog(@"同步請求測試開始...");
    NSString *urlString = @"http://www.yahoo.com";
    NSURL *url = [NSURL URLWithString:urlString];
    NSURLRequest *urlRequest = [NSURLRequest requestWithURL:url];
    NSURLResponse *response = nil;
    NSError *error = nil;
    NSLog(@"馬上進行同步連接請求url的數據");
    NSData *data = [NSURLConnection sendSynchronousRequest:urlRequest returningResponse:&response error:&error];
    if ([data length] > 0 && error == nil) {
        NSLog(@"%lu 字節的數據被返回.",(unsigned long)[data length]);
    }else if([data length] == 0 && error == nil){
        NSLog(@"沒有數據返回");
    }else if (error != nil){
        NSLog(@"出現錯誤= %@",error);
    }
    NSLog(@"同步方法測試完成."); //它會等待上面的下載數據完成才打印這句話
    
}

//使用GCD,同步加載數據.不會阻塞主線程。
-(void)fetchYahooData2_GCD{
    NSLog(@"使用GCD同步請求測試開始...");
    NSString *urlString = @"http://www.yahoo.com";
    NSLog(@"馬上進行同步連接請求url的數據...");
    dispatch_queue_t dispatchQueue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
    dispatch_async(dispatchQueue, ^{
        NSURL *url = [NSURL URLWithString:urlString];
        NSURLRequest *urlRequest = [NSURLRequest requestWithURL:url];
        NSURLResponse *response = nil;
        NSError *error = nil;
        NSData *data = [NSURLConnection sendSynchronousRequest:urlRequest returningResponse:&response error:&error];
        if ([data length] > 0 && error == nil) {
            NSLog(@"%lu 字節的數據被返回.",(unsigned long)[data length]);
//            [data writeToFile:@"/Users/macoslion/Desktop/new/download.html" atomically:YES]; //下載下來的數據保存為xml,或者html。最好是xml
            
        }else if ([data length] == 0 && error == nil){
            NSLog(@"沒有數據返回.");
        }else if (error != nil){
            NSLog(@"請求出錯 = %@",error);
        }
    });
    NSLog(@"GCD測試完成."); //它會直接打印,不會等到上面下載數據完成才打印
    
}


//使用異步,它也不會阻塞主線程
-(void)fetchYahooData3
{
    NSLog(@"異步請求測試開始..");
    NSString *urlString = @"http://www.yahoo.com";
    NSURL *url = [NSURL URLWithString:urlString];
    NSURLRequest *urlRequest = [NSURLRequest requestWithURL:url];

    NSLog(@"馬上進行異步連接請求url的數據...");
    
     
     [NSURLConnection sendAsynchronousRequest:urlRequest queue:[NSOperationQueue currentQueue] completionHandler:^(NSURLResponse *reponse, NSData *data, NSError *error) {
         if ([data length] > 0 && error == nil) {
             NSLog(@"%lu 字節的數據被返回.",(unsigned long)[data length]);
         }else if([data length] == 0 && error == nil){
             NSLog(@"沒有數據返回");
         }else if (error != nil){
             NSLog(@"請求出錯 = %@",error);
         }
     }];
   
    NSLog(@"異步方法測試完成"); //這句話不會等到上面打印完了而打印


}
@end


運行結果:

2014-07-22 18:46:03.744 NSURLConnectionDemo[2350:403] 同步請求測試開始...
2014-07-22 18:46:03.857 NSURLConnectionDemo[2350:403] 馬上進行同步連接請求url的數據
2014-07-22 18:46:06.229 NSURLConnectionDemo[2350:403] 361544 字節的數據被返回.
2014-07-22 18:46:06.229 NSURLConnectionDemo[2350:403] 同步方法測試完成.
2014-07-22 18:46:06.229 NSURLConnectionDemo[2350:403] -------------------------------------
2014-07-22 18:46:06.229 NSURLConnectionDemo[2350:403] 使用GCD同步請求測試開始...
2014-07-22 18:46:06.229 NSURLConnectionDemo[2350:403] 馬上進行同步連接請求url的數據...
2014-07-22 18:46:06.230 NSURLConnectionDemo[2350:403] GCD測試完成.
2014-07-22 18:46:06.230 NSURLConnectionDemo[2350:403] -------------------------------------
2014-07-22 18:46:06.230 NSURLConnectionDemo[2350:403] 異步請求測試開始..
2014-07-22 18:46:06.230 NSURLConnectionDemo[2350:403] 馬上進行異步連接請求url的數據...
2014-07-22 18:46:06.230 NSURLConnectionDemo[2350:403] 異步方法測試完成
2014-07-22 18:46:08.079 NSURLConnectionDemo[2350:1b03] 361556 字節的數據被返回.
2014-07-22 18:46:08.975 NSURLConnectionDemo[2350:403] 361547 字節的數據被返回.
下面兩個都沒有阻塞主程序,下面兩種方式得到字節流數據明顯是最後顯示的,一個結果是GCD使用同步的,一個是異步得到的結果。一個是也就是說沒有阻塞主線程。GCD和 異步分開測試看起來效果會更好。

上面異步加載的是使用類方法創建的異步請求得到的數據。還有一種使用實例變量方法去創建,並開啟,設置委托對象。這種方式更靈活。

異步創建對象:

- (id)initWithRequest:(NSURLRequest *)request delegate:(id < NSURLConnectionDelegate >)delegate

第二個參數就是設置委托的對象,也就是獲取數據的對象。

開啟請求:

- (void)start

取消異步加載請求:

- (void)cancel
如果你調用了這個方法,那麼設置委托的對象將不會再接收任何連接的消息了。如果想要重新連接,那麼就需要創建一個新的NSURLConnection對象了。

還可以異步創建並,設置委托對象,立馬啟動:

- (id)initWithRequest:(NSURLRequest *)request delegate:(id < NSURLConnectionDelegate >)delegate startImmediately:(BOOL)startImmediately
最後一個參數給YES,就會立馬啟動。給NO,那麼記得調用開啟請求的方法。

NSURLConnectionDelegate協議有一些必須要實現的方法,這些方法就放在設置委托的對象裡去執行。因為設置委托的對象遵循了這個協議。如下所示:

獲取 data流方法:

- (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data

這個方法在請求開始之後,間隔一段時間就會獲取data。因此要獲得全部data,要創建一個NSMutableData成員變量,再追加這個參數。才能獲取完整的data數據.


完成所有數據請求成功調用的方法:

- (void)connectionDidFinishLoading:(NSURLConnection *)connection


加載request出錯調用的方法,這個是可選的:

- (void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error

還有一些方法就不一一介紹了,詳細信息請參閱蘋果官方幫助文檔。如有錯誤,歡迎廣大博友指出,鄙人不慎感激。



  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved