NSURLConnection是苹果提供的原生网络访问类。下面这段话来自Objc对它的理解。
iOS 7 和 Mac OS X 10.9 Mavericks 中一个显著的变化就是对 Foundation URL 加载系统的彻底重构。
NSURLConnection 作为 Core Foundation / CFNetwork 框架的 API 之上的一个抽象,在 2003 年,随着第一版的 Safari 的发布就发布了。NSURLConnection 这个名字,实际上是指代的 Foundation 框架的 URL 加载系统中一系列有关联的组件:NSURLRequest、NSURLResponse、NSURLProtocol、 NSURLCache、 NSHTTPCookieStorage、NSURLCredentialStorage 以及同名类 NSURLConnection。
NSURLRequest 被传递给 NSURLConnection。被委托对象(遵守以前的非正式协议 <NSURLConnectionDelegate> 和 <NSURLConnectionDataDelegate>)异步地返回一个 NSURLResponse 以及包含服务器返回信息的 NSData。
在一个请求被发送到服务器之前,系统会先查询共享的缓存信息,然后根据策略(policy)以及可用性(availability)的不同,一个已经被缓存的响应可能会被立即返回。如果没有缓存的响应可用,则这个请求将根据我们指定的策略来缓存它的响应以便将来的请求可以使用。
在把请求发送给服务器的过程中,服务器可能会发出鉴权查询(authentication challenge),这可以由共享的 cookie 或机密存储(credential storage)来自动响应,或者由被委托对象来响应。发送中的请求也可以被注册的 NSURLProtocol 对象所拦截,以便在必要的时候无缝地改变其加载行为。
不管怎样,NSURLConnection 作为网络基础架构,已经服务了成千上万的 iOS 和 Mac OS 程序,并且做的还算相当不错。但是这些年,一些用例——尤其是在 iPhone 和 iPad 上面——已经对 NSURLConnection 的几个核心概念提出了挑战,让苹果有理由对它进行重构。
在 2013 的 WWDC 上,苹果推出了 NSURLConnection 的继任者:NSURLSession。
创建请求
NSURL *msgURL = [NSURL URLWithString:@"http://freezing-cloud-6077.herokuapp.com/messages.json"];
NSURLRequest *msgRequest = [NSURLRequest requestWithURL:msgURL cachePolicy:(NSURLRequestUseProtocolCachePolicy) timeoutInterval:60];
NSURLConnection *theConnection = [[NSURLConnection alloc] initWithRequest:msgRequest delegate:self];//默认自动开启start
if (theConnection) {
NSMutableData *connData = [[NSMutableData alloc] init];
[self setConnectionData:connData];
}else {
NSLog(@"Connection failed...");
}
要接收来自网络连接的数据,需要实现以下代理方法。
// 收到服务器响应
- (void)connection:(NSURLConnection *)connection didReceiveResponse:(nonnull NSURLResponse *)response {
[self.connectionData setLength:0];
}
// 将收到的数据附加到一个本地数据对象末尾
- (void)connection:(NSURLConnection *)connection didReceiveData:(nonnull NSData *)data {
[self.connectionData appendData:data];
}
- (void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error {
connection = nil;
self.connectionData = nil;
NSLog(@"Connection failed. Error - %@ %@",[error localizedDescription],[[error userInfo] objectForKey:NSURLErrorFailingURLStringErrorKey]);
}
- (void)connectionDidFinishLoading:(NSURLConnection *)connection {
NSString *retString = [NSString stringWithUTF8String:[self.connectionData bytes]];
NSLog(@"json data: %@", retString);
NSError *parseError = nil;
NSArray *jsonArray =
[NSJSONSerialization JSONObjectWithData:self.connectionData
options:0
error:&parseError];
if (!parseError) {
NSLog(@"json array is %@", jsonArray);
} else {
NSString *err = [parseError localizedDescription];
NSLog(@"Encountered error parsing: %@", err);
}
connection = nil;
self.connectionData = nil;
}
发送请求
NSURLConnection默认发送异步请求,使用步骤大致是:
1.创建NSURL对象。
2.传入NSURL创建一个NSURLRequest对象,设置请求头和请求体。
3.使用NSURLConnection发送NSURLRequest。
发送请求分为同步请求和异步请求:
异步请求
NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:messageBoardURL cachePolicy:NSURLRequestUseProtocolCachePolicy timeoutInterval:30.0];
[request setHTTPMethod:@"POST"];//请求方法
[request setValue:@"application/json" forHTTPHeaderField:@"Accept"];
[request setValue:@"application/json" forHTTPHeaderField:@"Content-Type"];//设置请求头和请求体
[request setValue:[NSString stringWithFormat:@"%lu",
(unsigned long)[jsonData length]] forHTTPHeaderField:@"Content-Length"]; //3
[request setHTTPBody: jsonData];
//异步请求
[NSURLConnection sendAsynchronousRequest:request queue:[[NSOperationQueue alloc]init] completionHandler:^(NSURLResponse * _Nullable response, NSData * _Nullable data, NSError * _Nullable connectionError) {
}];
同步请求
//1.确定请求路径
NSURL *url = [NSURL URLWithString:@""];
//2.创建一个请求对象(提供默认请求头信息,默认GET请求)
NSURLRequest *request = [NSURLRequest requestWithURL:url];
//3.把请求发送给服务器
NSHTTPURLResponse *response = nil;
NSError *error = nil;
NSData *data = [NSURLConnection sendSynchronousRequest:request returningResponse:&response error:&error];
//4.解析服务器返回的信息
NSString *jsonData = [[NSString alloc]initWithData:data encoding:NSUTF8StringEncoding];
与RunLoop关系
设置代理:
1.[[NSURLConnection alloc]initWithRequest:request delegate:self];
默认自动发送请求。
2.NSURLConnection *connect = [[NSURLConnection alloc]initWithRequest:request delegate:self startImmediately:NO];
当startImmediately是NO的时候不会发生请求,需要手动调用[connect start];
线程:
默认情况下代理方法在主线程进行,也可以手动设置队列
[connect setDelegateQueue:[[NSOperationQueue alloc]init]];
3.子线程中网络请求
//创建OperationQueue
NSOperationQueue *queue = [[NSOperationQueue alloc]init];
[queue addOperationWithBlock:^{
.....
//此方法内部会自动创建一个当前线程的runloop
[connection start];
//或者手动设置当前队列
[connection setDelegateQueue:[[NSOperationQueue alloc]init]];
[[NSRunLoop currentRunLoop]run];
}];