在将NSURLSession
与后台配置一起使用时,是否可以检测到DNS查找失败?
如果我使用默认配置和完成处理程序,则会在error参数中报告DNS解析失败。但是,如果我使用后台配置,则永远不会报告失败,并且从不调用委托方法。
NSURLSessionTaskDelegate
协议的Apple文档说:
不通过error参数报告服务器错误。唯一的 您的委托通过错误参数收到的错误 客户端错误 ,例如无法解析主机名 或 连接到主机。
这表明我应该在那里看到DNS失败消息。下面的代码说明了行为。我已经在iOS 8.4和9上对设备和模拟器进行了测试,并搜索了开发论坛,SO和使用流行的搜索引擎,但空洞。我敢肯定我一定会错过一些非常简单的事情。
import UIKit
@UIApplicationMain
class AppDelegate: UIResponder, UIApplicationDelegate, NSURLSessionTaskDelegate {
var window: UIWindow?
func application(application: UIApplication, didFinishLaunchingWithOptions launchOptions: [NSObject: AnyObject]?) -> Bool {
let bgsesh = NSURLSession(configuration: NSURLSessionConfiguration.backgroundSessionConfigurationWithIdentifier("foobarbazqwux"), delegate: self, delegateQueue: nil)
let dfsesh = NSURLSession(configuration: NSURLSessionConfiguration.defaultSessionConfiguration(), delegate: nil, delegateQueue: nil)
let url = NSURL(string: "http://foobarbaz.qwzx")!
let a = dfsesh.downloadTaskWithURL(url) { (url:NSURL?, resp:NSURLResponse?, err:NSError?) -> Void in
print(err)
/*
Optional(Error Domain=NSURLErrorDomain Code=-1003
"A server with the specified hostname could not be found."
UserInfo=0x146e8050 {
NSErrorFailingURLStringKey=http://foobarbaz.qwzx/,
_kCFStreamErrorCodeKey=8,
NSErrorFailingURLKey=http://foobarbaz.qwzx/,
NSLocalizedDescription=A server with the specified hostname could not be found.,
_kCFStreamErrorDomainKey=12,
NSUnderlyingError=0x14693ab0 "The operation couldn’t be completed. (kCFErrorDomainCFNetwork error -1003.)"
})
*/
}
a.resume()
let b = bgsesh.downloadTaskWithURL(url)
b.resume()
return true
}
func URLSession(session: NSURLSession, task: NSURLSessionTask, didCompleteWithError error: NSError?) {
// never runs, DNS failure is not reported for background config :-(
print("didCompleteWithError: \(task.taskIdentifier) \(error)")
}
// ... other delegate methods ...
}
答案 0 :(得分:2)
设置URLSessionConfiguration.timeoutIntervalForResource
。
资源超时间隔控制放弃之前等待整个资源传输的时间(以秒为单位)。
默认值为7天。
例如,
let conf = NSURLSessionConfiguration.backgroundSessionConfigurationWithIdentifier("foobarbazqwux")
conf.timeoutIntervalForResource = 60 // seconds
let session = NSURLSession(configuration: conf, delegate: self, delegateQueue: nil)
在这种情况下,如果60秒后仍然失败,系统将重试连接并调用委托。
在@algrid的评论之后进行编辑:
此值应考虑资源的传输时间。因此,它取决于下载/上传的资源大小和网络速度。