NSURLConnection同步和异步连接

来源:互联网 发布:用友成都java招聘 编辑:程序博客网 时间:2024/05/16 03:06

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数据.


       完成所有数据请求成功调用的方法:

<span style="font-family: Arial, Helvetica, sans-serif;">- (void)connectionDidFinishLoading:(NSURLConnection *)connection</span>


加载request出错调用的方法,这个是可选的:

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

还有一些方法就不一一介绍了,详细信息请参阅苹果官方帮助文档。如有错误,欢迎广大博友指出,鄙人不慎感激。



0 0
原创粉丝点击