1

我有一个用 Swift 1.2 编写的函数,用于检查地址或 IP 的可达性。这里是 :

func isHostConnected(hostAddress : String) -> Bool
{
    var response : NSURLResponse?
    let request = NSMutableURLRequest(URL: NSURL(string: hostAddress)!)
    request.timeoutInterval = 3

    let data = NSURLConnection.sendSynchronousRequest(request, returningResponse: &response, error: nil)

    return ((response as? NSHTTPURLResponse)!.statusCode == 200)
}

现在,NSURLConnection已弃用,根据 Xcode 的建议,我尝试使用 编写它NSURLSession.dataTaskWithRequest,这里是:

func isHostConnected(hostAddress : String) -> Bool
{
    let request = NSMutableURLRequest(URL: NSURL(string: hostAddress.stringByAddingPercentEscapesUsingEncoding(NSUTF8StringEncoding)!)!)
    request.timeoutInterval = 3

    let session = NSURLSession(configuration: NSURLSessionConfiguration.defaultSessionConfiguration(), delegate: nil, delegateQueue: NSOperationQueue.mainQueue())

    var responseCode = -1

    session.dataTaskWithRequest(request, completionHandler: {(data : NSData?, response : NSURLResponse?, error : NSError?) -> Void in
        responseCode = (response as? NSHTTPURLResponse)!.statusCode
    })!.resume()

    return (responseCode == 200)
}

上面的代码总是返回false(它的明显),因为completionHandler在单独的线程上执行。

我担心的是,我可以通过阻止它completionHandler()以某种方式在 MainThread 上运行吗?sendSynchronousRequest

我有理由不在这里使用“Apple's Reachabilty”。

任何建议都会有所帮助。:)

4

1 回答 1

11

(从https://stackoverflow.com/a/30670518/1187415重复我的论点:)

检查服务器上是否存在资源需要发送 HTTP 请求并接收响应。TCP 通信可能需要一些时间,例如,如果服务器很忙,客户端和服务器之间的某些路由器无法正常工作,网络中断等。

这就是为什么总是首选异步请求的原因。即使您认为请求应该只需要几毫秒,但由于某些网络问题,有时可能需要几秒钟。而且——众所周知——阻塞主线程几秒钟是一个很大的禁忌。

话虽如此,您可以使用“计数信号量”或“调度组”来等待某些异步任务的完成。你不应该主线程上使用它。阻塞主线程长达3秒是不能接受的!

func isHostConnected(hostAddress : String) -> Bool
{
    let request = NSMutableURLRequest(URL: NSURL(string: hostAddress.stringByAddingPercentEscapesUsingEncoding(NSUTF8StringEncoding)!)!)
    request.timeoutInterval = 3
    request.HTTPMethod = "HEAD"

    let session = NSURLSession(configuration: NSURLSessionConfiguration.defaultSessionConfiguration())
    var responseCode = -1

    let group = dispatch_group_create()
    dispatch_group_enter(group)

    session.dataTaskWithRequest(request, completionHandler: {(_, response, _) in
        if let httpResponse = response as? NSHTTPURLResponse {
            responseCode = httpResponse.statusCode
        }
        dispatch_group_leave(group)
    })!.resume()

    dispatch_group_wait(group, DISPATCH_TIME_FOREVER)
    return (responseCode == 200)
}

评论:

  • 将 HTTP 方法设置为“HEAD”是一个小的优化,因为服务器只发送响应头而不发送实际数据。
  • 在非法主机名的情况下,response将是nil,并且 responseCode = (response as? NSHTTPURLResponse)!.statusCode会崩溃。
于 2015-06-20T10:04:51.430 回答