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

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

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

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

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

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

时间: 2024-10-12 07:17:12

NSURLConnection同步和异步连接的相关文章

[IOS_HTTP]NSURLConnection同步与异步请求

今天看到<WIN32多线程程序设计>的同步控制时,才发现原来自己对同步和异步的概念很模糊,甚至混淆.于是GOOGLE了一下.下面都是高人们的见解,简单明了. 同步是指:发送方发出数据后,等接收方发回响应以后才发下一个数据包的通讯方式.  异步是指:发送方发出数据后,不等接收方发回响应,接着发送下个数据包的通讯方式. 举个不太恰当的例子,就像:  SendMessage(...)  TRACE0("just  like  send");   PostMessage(...) 

iOS网络编程--NSConnection的同步连接与异步连接

1 // 2 // ZFViewController.m 3 // 0628-表单验证 4 // 5 // Created by zfan on 14-6-28. 6 // Copyright (c) 2014年 zfan. All rights reserved. 7 // 8 9 #import "ZFViewController.h" 10 #import "MBProgressHUD+MJ.h" 11 12 @interface ZFViewControll

TCP同步与异步,长连接与短连接【转载】

原文地址:TCP同步与异步,长连接与短连接作者:1984346023 [转载说明:http://zjj1211.blog.51cto.com/1812544/373896   这是今天看到的一篇讲到TCP同步与异步,长连接与短连接编程的文章,内容不多,却把概念将的比较清晰.] 标签:TCP 线程 异步 模式 阻塞 首先我简单介绍一下同步TCP编程 与异步TCP编程. 在服务端我们通常用一个TcpListener来监听一个IP和端口.客户端来一个请求的连接,在服务端可以用同步的方式来接收,也可以用

Adobe AIR中使用Flex连接Sqlite数据库(1)(创建数据库和表,以及同步和异步执行模式)

系列文章导航 Adobe AIR中使用Flex连接Sqlite数据库(1)(创建数据库和表) Adobe AIR中使用Flex连接Sqlite数据库(2)(添加,删除,修改以及语句参数) Adobe AIR中使用Flex连接Sqlite数据库(3)(查询) Adobe AIR中使用Flex连接Sqlite数据库(4)(事务) Flex,Fms3相关文章索引 Fms3和Flex打造在线多人视频会议和视频聊天(附原代码) 免费美女视频聊天,多人视频会议功能加强版本(Fms3和Flex开发(附源码))

WCF的同步和异步(以WPF连接为例)

2016-06-0711:05:44 在学习WCF时,学到WCF服务的同步和异步. 我理解的同步是: 当WCF服务是同步执行时,程序只有一条线程,代码只能按顺序一步一步来执行,当执行客户端/服务端某方法需要10秒时,只能等待10秒才能接着执行之后的代码. 当WCF服务是异步执行时,程序可以同时存在多条线程同时运行,当客户端/服务端某方法执行10秒的同时,之后的代码也在执行操作. 1.先建立一个WCF服务和WPF项目,WPF项目引用WCF服务,配置好WPF项目的WCF服务引用 2.服务端接口:IT

同步和异步请求

// //  ViewController.m //  UI-NO-18 // //  Created by Bruce on 15/8/13. //  Copyright (c) 2015年 Bruce. All rights reserved. // #import "ViewController.h" #import "InfoModel.h" @interface ViewController () @end @implementation ViewCont

iOS中的请求(GET请求,POST请求,同步请求,异步请求)

1.用到的一些第三方 PostTableViewController.m #import "PostTableViewController.h" @interface PostTableViewController () @property(nonatomic,retain) NSArray *dataArr; @end @implementation PostTableViewController - (void)viewDidLoad { [super viewDidLoad];

iosGET同步、异步请求

一.同步GET 1 2 3 4 5 6 7 8 9 10 11 12 // 1.将网址初始化成一个OC字符串对象 NSString *urlStr = [NSString stringWithFormat:@"%@?query=%@?ion=%@&output=json&ak=6E823f587c95f0148c19993539b99295", kBusinessInfoURL, @"银行", @"济南"]; // 如果网址中存在

网络请求数据 get请求方式 &nbsp; post请求 协议异步连接服务器 block异步连接服务器

网络请求三部 创建一个请求(添加接口,对接口进行解码,) 设定请求方式(将接口转为NSURL,设置请求[请求地址, 缓存策略, 超时时间],设置请求方式) 连接服务器([同步连接,异步连接]代理连接,block连接) #import "MainViewController.h" @interface MainViewController () @property (retain, nonatomic) IBOutlet UIImageView *ImageWiew; //get请求方法