以 Objective-C 中创建 NSURLSessionDownloadTask
为例:
NSURLSessionDownloadTask *task = [[NSURLSession sharedSession] downloadTaskWithURL:[NSURL URLWithString:@"google.com"] completionHandler:^(NSURL *location, NSURLResponse *response, NSError *error) {
if (task.state == NSURLSessionTaskStateCompleted) {
// Do things
}
}];
[task resume];
我也可以在完成 block 中毫无问题地访问我正在创建的任务,task
。
但是,在 Swift 中,如果我尝试同样的事情:
let URL = NSURL(string: "google.com")
let task = NSURLSession.sharedSession().downloadTaskWithURL(URL, completionHandler: { location, response, error in
if task.state == .Completed {
// Do things
}
}
task.resume()
我可能会出现“变量正在其自身初始值内使用”的错误。
我该如何规避这个问题?
最佳答案
更新答案 我现在意识到我没有仔细阅读你的代码。您没有将闭包传递给初始值设定项,而是传递给方法。当将闭包传递给初始值设定项时,我最初编写的内容仍然有效,但在您的情况下则不然。
不过你的问题很相似。
您有一个 task
变量,它使用函数的返回值进行初始化。您将一个闭包传递给该函数,并在该闭包内引用 task
变量。
编译器不知道闭包何时执行(至少我不认为它会检查这一点,这是一个 downloadTaskWithURL()
内部实现细节) - 并且有可能是在函数体中执行(而不是将其存储在属性中并在稍后执行)。如果闭包在函数体中执行,那么它会在尚未赋值时访问 task
变量(因为函数仍在执行)。
如果有一种方法让编译器知道闭包没有在函数体中执行,那么编译器就有可能处理这种情况。但 swift 并没有实现类似的东西。
结论:我很欣赏编译器为此抛出错误,因为否则我会期望出现运行时异常 - 尽管可能不是在您的特定情况下(因为闭包稍后执行)。
原始答案
正如您可能知道的,在 swift 中,只有正确初始化所有类/结构属性并且调用基本初始化程序(对于继承类),self
才可用
您将闭包传递给类初始值设定项 - 编译器无法确定何时执行闭包,因此闭包本身不能包含对 self
的任何(直接或间接)引用。
在你的例子中,task
是被实例化的变量,所以当你在闭包中使用它时,就像你在使用self
。这在 swift 中是不允许的,所以你会遇到这个错误。
在 Objective C 中不会发生同样的情况,因为初始化器中没有这样的约束。
但请注意,从概念上讲,您所做的事情看起来并不是一个好的做法。您正在读取类实例在正确初始化之前的属性。要确定调用的状态,您应该依赖传递给闭包的参数,理想情况下它应该提供您需要的所有信息。
关于objective-c - 检查初始化方法的完成 block 内 Swift 变量的值,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26407349/