NSURLConnection委托方法未被调用

时间:2012-06-20 19:14:55

标签: objective-c ios ssl https

我正在尝试使用NSURLConnection的委托方法。 目前尚未调用以下方法:

- (void)connectionDidFinishLoading:(NSURLConnection *)connection;

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

- (void)connection:(NSURLConnection *)connection didReceiveAuthenticationChallenge:(NSURLAuthenticationChallenge *)challenge;
//I also want to be able to use self-signed https urls

- (BOOL)connection:(NSURLConnection *)connection canAuthenticateAgainstProtectionSpace:(NSURLProtectionSpace *)protectionSpace;
//I also want to be able to use self-signed https urls

我目前正在使用同步调用,但异步似乎更好,因为在我完成代码库后,我将把它实现到iPhone应用程序中,我不能让我的ui冻结。

使用以下方法responseData = [NSMutableData dataWithData:[NSURLConnection sendSynchronousRequest:request returningResponse:&response error:&error]];

我收回了我需要的数据但是使用异步我似乎必须使用委托方法来获取数据。我尝试使用@interface myClass : NSObject<NSURLConnectionDelegate>

添加委托

我正在调用我的方法如下:

-(void)grabData{
    NSArray* array = [NSArray arrayWithObjects:@"auth.login",@"user",@"pass", nil];
    NSData* packed_array = [array messagePack];

    NSURL* url = [NSURL URLWithString:@"https://192.168.1.115:3790/"];
    NSMutableURLRequest* request = [[[NSMutableURLRequest alloc]initWithURL:url]retain];
    [request setHTTPMethod:@"POST"];
    [request setValue:@"RPC Server" forHTTPHeaderField:@"Host"];
    [request setValue:@"binary/message-pack" forHTTPHeaderField:@"Content-Type"];
    [request setValue:[NSString stringWithFormat:@"%d",[packed_array length]] forHTTPHeaderField:@"Content-Length"];
    [request setHTTPBody:packed_array];

    //NSHTTPURLResponse *response = nil;
    NSError *error = nil;

    NSLog(@"connecting");
    NSURLConnection* connection = [[[NSURLConnection alloc]initWithRequest:request delegate:self]retain];
    if (connection) {
        NSLog(@"connection exists");
        self.responseData = [[NSMutableData data]retain];
    }
    else {
        NSLog(@"Connection doesn't exist?");
    }
    NSLog(@"response data: %@",[responseData messagePackParse]);
    NSLog(@"error: %@",error);
}

我尝试了以下内容:

NSURLConnection *connection = [[NSURLConnection alloc] initWithRequest:request delegate:self];
[connection start];

5 个答案:

答案 0 :(得分:9)

我继续搜索相关问题并找到this answer most helpful。它引导我使用以下方法。 首先,我在我的界面中声明了一个名为finished的BOOL属性,并在我的实现中添加了以下方法,这导致我的委托方法被调用。

while(!finished) {
    [[NSRunLoop currentRunLoop] runMode:NSDefaultRunLoopMode beforeDate:[NSDate distantFuture]];
}

答案 1 :(得分:2)

只是一个想法 - 你使用自动引用计数的任何机会,并且在你调用[连接开始]和应该调用委托方法之间释放创建NSURLConnection的对象(即数据已经被释放)接收到的)?

是否有任何内容维护对您用于创建NSURLConnection的对象的引用?

根据其他信息,这似乎是一个可能的原因。您可能在App Delegate(或您创建的其他类)中有以下代码。

MyClass *object = [[MyClass alloc] init];
[object grabData];

这很棒 - 它实例化一个对象,然后告诉该对象获取有问题的数据。但是,一旦grabData完成(并且在NSURLConnection返回数据之前),对象将被释放,因为没有任何东西保留它。解决这个问题:

  1. 在.h文件中创建一个属性(您创建对象的位置)以保留您的MyClass实例。

    @property(strong)MyClass * object;

  2. 合成.m文件中的该属性

    @synthesize object;

  3. 不要只是创建对象,而是保持对它的引用:

    MyClass * myClass = [[MyClass alloc] init];

    [myClass grabData];

    self.object = myClass;

  4. 这将阻止您的对象被ARC取消分配。

    当您完成对象(您的NSURLConnection已返回数据)时,您可以设置self.object = nil来摆脱该引用。

答案 2 :(得分:1)

您可以使用:

NSURLConnection *connection = [[NSURLConnection alloc] initWithRequest: request delegate:self startImmediately:NO];

[connection scheduleInRunLoop:[NSRunLoop mainRunLoop] forMode:NSDefaultRunLoopMode];

[connection start];

答案 3 :(得分:0)

如果您遇到问题,您可以执行我过去所做的工作,并使用同步请求,如下所示:

NSHTTPURLResponse *response = nil;
NSError *error = nil;
[NSURLConnection sendSynchronousRequest:request returningResponse:&response error:&error]

这将输出您需要的数据,所有您需要做的就是不要锁定您的UI是使用块(GCD):

dispatch_queue_t downloadQueue = dispatch_queue_create("download queue", NULL);
dispatch_async(downloadQueue, ^{
  NSHTTPURLResponse *response = nil;
  NSError *error = nil;
  NSData *myData = [NSURLConnection sendSynchronousRequest:request returningResponse:&response error:&error]
  dispatch_async(dispatch_get_main_queue(), ^{
    // Handle your response data
  });
});

答案 4 :(得分:0)

建立连接后,请写下面

[连接开始]; 你的委托方法将被调用。