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
另一些方法就不一一介绍了。具体信息请參阅苹果官方帮助文档。
如有错误。欢迎广大博友指出。鄙人不慎感激。