objective-c - 检查初始化方法的完成 block 内 Swift 变量的值

标签 objective-c swift closures objective-c-blocks

以 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/

相关文章:

closures - "to close over the enclosing scope/class"是什么意思?

swift - 从闭包调用的方法中的 self 上下文

ios - 完成处理程序闭包是否始终在后台线程中运行?

ios - VC之间传输数据

ios - 在哪里访问 IBOutlet var 并使用它来定义类可访问的常量

ios - 如何让听众在离线时在 Firebase 中工作以进行多位置更新

ios - 在 Swift 中将图像合并为 PDF

objective-c - NSTableview刷新异步加载数据

ios - Objective-C 编译器遗漏了协议(protocol)定义

objective-c - Uiswitch开/关