Ios NSURL连接超时
在我们的iPhone应用程序中,NSURLConnection请求超时的问题时有发生。最近似乎发生得更多。一旦它进入这种状态,它就会保持这种状态。唯一的解决方案似乎是关闭应用程序并重新启动它 意见:Ios NSURL连接超时,ios,nsurlconnection,Ios,Nsurlconnection,在我们的iPhone应用程序中,NSURLConnection请求超时的问题时有发生。最近似乎发生得更多。一旦它进入这种状态,它就会保持这种状态。唯一的解决方案似乎是关闭应用程序并重新启动它 意见: 执行NSURLConnection的核心代码没有更改(最近添加的一些自定义用户代理代码除外) 目前还没有找到一个可复制的例子,但在应用程序在后台运行一段时间后,似乎会出现超时,特别是在3G(无WiFi)上运行时 服务器上的Apache在遇到这些超时时未记录来自客户端的任何请求 一些迹象表明,其他应
- 执行NSURLConnection的核心代码没有更改(最近添加的一些自定义用户代理代码除外)李>
- 目前还没有找到一个可复制的例子,但在应用程序在后台运行一段时间后,似乎会出现超时,特别是在3G(无WiFi)上运行时
- 服务器上的Apache在遇到这些超时时未记录来自客户端的任何请求李>
- 一些迹象表明,其他应用程序(如Mail和Safari)会受到影响(即出现超时),但并不一致
- 我所处的3G覆盖率很稳定,不排除引发问题的暂时性问题(假设不太可能)
- 所有请求都将发送到我们自己的API服务器,并且都是restful POST请求
- 由于timeoutInterval和POST请求的问题,我们使用自己的基于NSTimer的超时。我尝试过增加超时值——问题仍然存在
- 应用程序最近已转换为ARC
- 在iOS 5.1.1下运行应用程序
- 该应用程序使用最新版本的UrbanAirship、TestFlight和Flurry SDK
- 还使用TouchXML的ARC分支解析响应
#define relnil(v) (v = nil)
- (id) initWebRequestController
{
self = [super init];
if (self)
{
//setup a queue to execute all web requests on synchronously
dispatch_queue_t aQueue = dispatch_queue_create("com.myapp.webqueue", NULL);
[self setWebQueue:aQueue];
}
return self;
}
- (void) getStuffFromServer
{
dispatch_queue_t aQueue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
dispatch_async(aQueue, ^{
dispatch_sync([self webQueue], ^{
error_block_t errorBlock = ^(MyAppAPIStatusCode code, NSError * error){
dispatch_async(dispatch_get_main_queue(), ^{
[[self delegate] webRequestController:self didEncounterErrorGettingPointsWithCode:code andOptionalError:error];
});
};
parsing_block_t parsingBlock = ^(CXMLDocument * doc, error_block_t errorHandler){
NSError * error = nil;
CXMLNode * node = [doc nodeForXPath:@"apiResult/data/stuff" error:&error];
if (error || !node) {
errorHandler(MyAppAPIStatusCodeFailedToParse, error);
}
else {
stuffString = [node stringValue];
}
if (stuffString) {
dispatch_async(dispatch_get_main_queue(), ^{
[[self delegate] webRequestController:self didFinishGettingStuff:stuffString];
});
}
else {
errorHandler(MyAppAPIStatusCodeFailedToParse, error);
}
};
NSURL * url = [[NSURL alloc] initWithString:[NSString stringWithFormat:MyAppURLFormat_MyAppAPI, @"stuff/getStuff"]];
NSMutableURLRequest * urlRequest = [[NSMutableURLRequest alloc] initWithURL:url];
NSMutableDictionary * postDictionary = [NSMutableDictionary dictionaryWithObjectsAndKeys:
[[NSUserDefaults standardUserDefaults] objectForKey:MyAppKey_Token], @"token",
origin, @"from",
destination, @"to",
transitTypeString, @"mode",
time, @"time",
nil];
NSString * postString = [WebRequestController httpBodyFromDictionary:postDictionary];
[urlRequest setHTTPBody:[postString dataUsingEncoding:NSUTF8StringEncoding]];
[urlRequest setHTTPMethod:@"POST"];
if (urlRequest)
{
[self performAPIRequest:urlRequest withRequestParameters:postDictionary parsing:parsingBlock errorHandling:errorBlock timeout:kTimeout_Standard];
}
else
{
errorBlock(MyAppAPIStatusCodeInvalidRequest, nil);
}
relnil(url);
relnil(urlRequest);
});
});
}
- (void) performAPIRequest: (NSMutableURLRequest *) request
withRequestParameters: (NSMutableDictionary *) requestParameters
parsing: (parsing_block_t) parsingBlock
errorHandling: (error_block_t) errorBlock
timeout: (NSTimeInterval) timeout
{
NSAssert([self apiConnection] == nil, @"Requesting before previous request has completed");
NSString * postString = [WebRequestController httpBodyFromDictionary:requestParameters];
[request setHTTPBody:[postString dataUsingEncoding:NSUTF8StringEncoding]];
NSString * erVersion = [[[NSBundle mainBundle] infoDictionary] objectForKey:@"CFBundleShortVersionString"];
NSString * erBuildVersion = [[[NSBundle mainBundle] infoDictionary] objectForKey:@"CFBundleVersion"];
if ([erBuildVersion isEqualToString:erVersion] || [erBuildVersion isEqualToString:@""]) {
erBuildVersion = @"";
} else {
erBuildVersion = [NSString stringWithFormat:@"(%@)", erBuildVersion];
}
NSString * iosVersion = [[UIDevice currentDevice] systemVersion];
NSString * userAgent = [NSString stringWithFormat:@"MyApp/%@%@ iOS/%@", erVersion, erBuildVersion, iosVersion];
[request setValue:userAgent forHTTPHeaderField:@"User-Agent"];
[request setTimeoutInterval:(timeout-3.0f)];
dispatch_sync(dispatch_get_main_queue(), ^{
NSURLConnection * urlConnection = [[NSURLConnection alloc] initWithRequest:request delegate:self startImmediately:NO];
if (urlConnection)
{
[self setApiConnection:urlConnection];
requestParseBlock = [parsingBlock copy];
requestErrorBlock = [errorBlock copy];
NSMutableData * aMutableData = [[NSMutableData alloc] init];
[self setReceivedData:aMutableData];
relnil(aMutableData);
[urlConnection scheduleInRunLoop:[NSRunLoop currentRunLoop] forMode:NSRunLoopCommonModes];
[urlConnection start];
relnil(urlConnection);
NSTimer * aTimer = [NSTimer scheduledTimerWithTimeInterval:timeout target:self selector:@selector(timeoutTimerFired:) userInfo:nil repeats:NO];
[self setTimeoutTimer:aTimer];
}
else
{
errorBlock(MyAppAPIStatusCodeInvalidRequest, nil);
}
});
//we want the web requests to appear synchronous from outside of this interface
while ([self apiConnection] != nil)
{
[NSThread sleepForTimeInterval:.25];
}
}
- (void) timeoutTimerFired: (NSTimer *) timer
{
[[self apiConnection] cancel];
relnil(apiConnection);
relnil(receivedData);
[self requestErrorBlock](MyAppAPIStatusCodeTimeout, nil);
requestErrorBlock = nil;
requestParseBlock = nil;
}
- (void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error
{
[self requestErrorBlock](MyAppAPIStatusCodeFailedToConnect, error);
relnil(apiConnection);
relnil(receivedData);
[[self timeoutTimer] invalidate];
relnil(timeoutTimer);
requestErrorBlock = nil;
requestParseBlock = nil;
}
- (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response
{
[receivedData setLength:0];
}
- (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data
{
[receivedData appendData:data];
}
- (void)connectionDidFinishLoading:(NSURLConnection *)connection
{
MyAppAPIStatusCode status = MyAppAPIStatusCodeFailedToParse;
CXMLDocument *doc = [[self receivedData] length] ? [[CXMLDocument alloc] initWithData:[self receivedData] options:0 error:nil] : nil;
DLog(@"response:\n%@", doc);
if (doc)
{
NSError * error = nil;
CXMLNode * node = [doc nodeForXPath:@"apiResult/result" error:&error];
if (!error && node)
{
status = [[node stringValue] intValue];
if (status == MyAppAPIStatusCodeOK)
{
[self requestParseBlock](doc, [self requestErrorBlock]);
}
else if (status == MyAppAPIStatusCodeTokenMissingInvalidOrExpired)
{
[Definitions setToken:nil];
[self requestMyAppTokenIfNotPresent];
[Definitions logout];
dispatch_async(dispatch_get_main_queue(), ^{
[[self delegate] webRequestControllerDidRecivedExpiredTokenError:self];
});
}
else
{
[self requestErrorBlock](status, nil);
}
}
else
{
[self requestErrorBlock](status, nil);
}
}
else
{
status = MyAppAPIStatusCodeUnexpectedResponse;
[self requestErrorBlock](status, nil);
}
relnil(doc);
relnil(apiConnection);
relnil(receivedData);
[[self timeoutTimer] invalidate];
relnil(timeoutTimer);
requestErrorBlock = nil;
requestParseBlock = nil;
}
下面的URL是应用程序处于问题状态时队列/线程的一些屏幕截图。注意,我相信线程10与前一个超时执行的取消有关,尽管互斥等待很奇怪。此外,线程22中关于Flurry的部分在其他情况下遇到问题时并不总是出现
堆栈跟踪屏幕截图:
也许我忽略了这些痕迹中的一些明显错误,因为我对iOS/Apple开发相对较新
如果我有NSURLConnection和相关代码的源代码,所有这些问题都会更容易解决,但事实就是如此,我在这一点上是在暗中摸索。删除TestFlight 1.0 SDK似乎可以解决问题。TestFlight还确认他们正在进行修复。考虑到这个bug最初被其他人确认已经一个多月了,我想知道我们离修复有多近 您发布的信息太多,但可能还不够。什么是错误消息?没有错误消息-就是这样。NSURLConnection启动,但从未完成,我们的NSTimer启动取消它。在想到我到处都在搜索类似的问题后,我偶然发现:我想这可能是我悲伤的原因。将尝试删除TestFlight,看看会发生什么。在不知道效果的情况下滥用dispatch lib。实际上,您的代码绑定了两个次要线程,它们只是等待主线程上的工作完成。这毫无意义。我想我在堆栈跟踪中发现了一些具有互斥的东西。。。线程10中的TfRunLoop操作是TestFlight代码。关于此TestFlight错误的进一步讨论如下: