从服务器检索数据时,闭包究竟做了什么?

What exactly does a closure do when retrieving data from a server?

我看了一个关于闭包的视频,有人用这种方式演示了闭包的基础知识:

func outer(howMuch: Int) -> () -> Int {
    var total = 0

    inner() {
        howMuch += total
        return total
    }
return inner
}

他接着说,当你这样做的时候:

let incrementBy10 = outer(10)

他说 incrementBy10 引用了闭包内部的 inner() 函数。

然后他继续进行一个检索数据的实际例子:

let url = "*url here*"
let nsURL = NSURLSession.shareSession().dataTaskWithUrl(nsURL) {(data,response,error) in 
print(NSString(data: data, encoding: NSUTF8StringEncoding)) }

'incrementby10' 示例与从服务器获取一些数据的实际示例有何关系。我没看懂他的意思:"when you grab something from a url, you are not gonna have the content immediately. You can call the closure when the url has been downloaded."

这是一个异步回调的例子。

异步回调用于在长运行 操作(例如网络请求)完成时执行闭包。它们允许我们触发网络请求,传入回调,然后在网络操作进行时继续执行其他代码。只有当操作完成时,才会执行闭包,并将服务器返回的数据作为参数传入。

如果我们不使用异步闭包,当我们从服务器获取内容时,应用程序会冻结(执行会停止)。这将是一个同步网络请求,它不会被使用,因为它会导致非常缓慢 UI 和糟糕的用户体验。

NSURLSessiondataTaskWithURL 本质上是一个异步 API,它接受一个闭包作为参数并在收到响应时触发它。


异步回调

异步回调网络调用示例(将其添加到 Swift 游乐场):

import UIKit
import XCPlayground // Only needed for Playground

// Only needed for Playground
XCPlaygroundPage.currentPage.needsIndefiniteExecution = true

class HTTP {

    class func GET(onSuccess: NSData -> Void ) {

        NSURLSession.sharedSession().dataTaskWithURL(NSURL(string: "http://httpbin.org/get")!, completionHandler: { data, response, error in
            onSuccess(data!)

        }).resume()
    }
}


print("About to fire request")
HTTP.GET({ payload in
    let response = NSString(data: payload, encoding: NSUTF8StringEncoding)
    print("Got network response: \(response)")
})
print("Just fired request")

打印的结果不是您直观预期的结果:

About to fire request
Just fired request
Got network response: ...

Just fired request在之前打印Got network response: ...因为网络请求是异步执行的。

上述代码的同步版本将产生以下输出:

About to fire request
Got network response: ...
Just fired request