我最近一直在努力解决HTTP超时问题。经过一个多月的调查,我很确定它是由错误的HTTP持久连接引起的。详情如下:
NSURLConnection
。keep alive
bug,但我的是另一个问题。更具体地说,该错误导致NSURLErrorNetworkConnectionLost
,但我的错误是NSURLErrorTimedOut
。但是,我不确定我的问题是否是由iOS 8的另一个错误造成的。NSURLErrorTimedOut
,以及所有跟随(距离最后一个请求不太远)重用持久连接请求的时间会导致NSURLErrorTimedOut
。NSURLErrorTimedOut
失败。从解决方法中我们可以看到所有这些都有效,因为它们会导致丢弃不良的持久连接并创建新的持久连接。我的问题:
NSURLConnection
不重复使用当前的持久连接,而是创建一个新连接,以便在我的代码中检测到它后可以解决这个问题?我通过使用CFNetwork
并直接控制Connection
标头成功地在iOS 8上缓解了此问题。然而,在iOS 9上似乎问题变得更糟。
由于我希望Apple能够在iOS 9上修复它,我终于开了一个雷达:http://www.openradar.me/22770738。
如果你也遇到这个问题,请复制我的雷达,或者更好的是,如果你有一个更可靠的可重复样本,你可以发射你自己的雷达。
答案 0 :(得分:2)
经过2周的研究,我可以回答问题3和4:
nginx
的持久连接超时在服务器上设置为5秒,这不应该是原因。服务器工程师发现这些超时请求实际上是正常接收和响应的。所以它更可能是客户端问题。由于我有一个minimal reproducible code来排除我的代码作为原因,因此原因应该在iOS中。CFNetwork
。更高级别的API(例如NSURLConnection
或NSURLSession
的{{1}}标头将被系统覆盖。 答案 1 :(得分:0)
同样的问题,iOS只是在服务器丢弃后尝试重用连接。
大约两年前,我转而使用CFNetwork来解决这个问题,但最近我发现用CFNetwork API实现SSL固定是不可能的。所以现在我正在考虑回到NSURLSession。
经过一番挖掘,我发现系统将不重用NSURLSession
之间的连接,因此在一段时间内创建新会话应该可以解决问题。
但我也发现(至少在macOS上):NSURLSession建立的每个连接都可以持续持续180秒,并且该连接没有通过释放关闭或重置会话,因此您可能需要实现一些缓存机制避免同时创建大量连接。
以下是我目前使用的简单机制:
@interface HTTPSession : NSObject
@property (nonatomic, strong) NSURLSession * urlSession;
@property (nonatomic, assign) CFTimeInterval flushTime;
@end
+ (NSURLSession *)reuseOrCreateSession
{
static NSMutableArray<HTTPSession *> * sessions = nil;
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
sessions = [NSMutableArray<HTTPSession *> array];
});
const CFTimeInterval serverTimeoutSeconds = 10;
const CFTimeInterval systemTimeoutSeconds = 40;
CFTimeInterval now = CFAbsoluteTimeGetCurrent();
HTTPSession * resultSession = nil;
for (HTTPSession * session in sessions) {
CFTimeInterval lifeTime = now - session.flushTime;
if (lifeTime < serverTimeoutSeconds) {
resultSession = session;
break;
}
if (lifeTime > systemTimeoutSeconds) {
resultSession = session;
resultSession.flushTime = now;
break;
}
}
if (!resultSession) {
resultSession = [HTTPSession new];
NSURLSession * session = [NSURLSession sessionWithConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration]];
// setup session
resultSession.urlSession = session;
resultSession.flushTime = now;
[sessions addObject:resultSession];
}
return resultSession.urlSession;
}
答案 2 :(得分:-1)
如果您为所有请求网址添加时间戳,该怎么办?我认为这会使每个请求都是唯一的,每次发送请求时iOS都会建立新的连接(我不确定。需要尝试)