NSURLResponse运行并发下载时实际数据的内容长度不匹配大小

时间:2013-03-27 11:52:35

标签: asynchronous nsurlconnection nsurlrequest content-length simultaneous-calls

我有一个UITableView,当选择一个单元格时,我进行服务调用以从Web服务异步下载PDF文件。它工作得很好,直到您一个接一个地(单独地)选择多个单元格,然后事情开始向南...

以下是一些(摘录)代码以澄清:

在MasterViewController.m中:

- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath
{
    //To uniquely identify this field, I build a simple string using the indexPath.
    NSString *key = [[NSString alloc]initWithFormat:@"%d.%d",pIndex.section,pIndex.row];

    //Use a pre-populated NSDictionary to specify the file I want from the server.
    NSString *reportID = [reportDictionary valueForKey:key];

    NSURL *url = [NSURL URLWithString:[NSString stringWithFormat:@"%@/RequestReportWithID",myConnectionObject.ServiceURL]];

    NSMutableDictionary *body = [[NSMutableDictionary alloc] initWithObjectsAndKeys:reportID, @"reportID", nil];

    NSData *requestData = [NSJSONSerialization dataWithJSONObject:dic options:0 error:nil];

    NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];

    [request setHTTPMethod:@"POST"];
    [request setValue:@"application/json" forHTTPHeaderField:@"Content-Type"];
    [request setValue:[NSString stringWithFormat:@"%d", [requestData length]] forHTTPHeaderField:@"Content-Length"];
    [request setHTTPBody: requestData];

    //Create a new object as delegate to receive the data, also add the key to assist with identification.
    ReportDownloader *newReportDownloader = [[ReportDownloader alloc]initWithDelegate:self andKey:key];

    NSURLConnection *connection = [[NSURLConnection alloc]initWithRequest:request delegate:newReportDownloader];
    if (connection)
    {
        //Nothing to do here, request was made.
    }
}

在ReportDownloader.m中:

long long expectedReportSize;
NSString *key;
NSData *thisReport;

-(NSObject*)initWithDelegate:(NSObject*)pDelegate andKey:(NSString*)pKey
{
    myTypedDelegate = (MasterViewController*)pDelegate;
    Key = pKey;
    return self;
}

- (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response {
    expectedReportSize = [response expectedContentLength];
    NSLog(@"Key:%@, Expected Size=%lld bytes",Key, [response expectedContentLength]);
    thisReport = [[NSMutableData alloc] init];
}

- (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data 
{
    [thisReport appendData:data];
}

- (void)connectionDidFinishLoading:(NSURLConnection *)connection 
{
    //Here is where things go wrong...
    NSLog(@"Key:%@, Actual Size=%ld bytes",Key, (unsigned long)[thisReport length]);
}

上述代码适用于单个报告。在快速线上,它适用于3-4个报告,但尝试做更多,并且预期长度开始与实际长度不匹配。我无法理解为什么你......

查看上面代码生成的输出: (请注意,只正确下载了indexPath 1.1的报告)

Key:1.0, Expected Size=304006 bytes
Key:1.3, Expected Size=124922 bytes
Key:1.0,   Actual Size=369494 bytes
Key:1.3,   Actual Size=380030 bytes
Key:1.2, Expected Size=179840 bytes
Key:1.4, Expected Size=377046 bytes
Key:1.2,   Actual Size=114376 bytes
Key:1.5, Expected Size=175633 bytes
Key:1.4,   Actual Size=180558 bytes
Key:1.5,   Actual Size=274549 bytes
Key:1.1, Expected Size=443135 bytes
Key:1.1,   Actual Size=443135 bytes

尝试打开PDF文档时会确认数据损坏,对于所有文件的数据大小与expectedContentLength不匹配,该文档都会失败。

我对这种行为感到非常困惑,我希望我犯了一个明显的n00b错误,并且有人在这里发现它。

为什么我制作ReportDownloader类的主要原因是为了避免像这样的问题。

提前感谢任何花时间扫描我的代码的人!

1 个答案:

答案 0 :(得分:0)

好的,这个问题的解决方案可以在这里找到: https://stackoverflow.com/a/1938395/1014983 詹姆斯·沃尔德的回答虽然不是问题的答案,但却完全解决了多个同步异步NSURLC连接的问题。

如果您需要一些实施方面的帮助,请告诉我,但这很简单。