导致“常量在初始化前被闭包捕获的原因”错误

问题描述 投票:7回答:1

在下一堂课中

class Foo {
   let _defaultValue = "N/A"
   let value: String 

   init (dict: NSDictionary) {
       self.value = dict["bar"] as? String! ?? _defaultValue
   }
}

编译器失败,并显示以下消息constant 'self.value' captured by a closure before being initialized

据我所知,没有运算符读取`self.value。该消息似乎有些混乱。

我不小心想到了一种解决方法。我应该说这让我更加困惑:

class Foo {
       let value: String 

       init (dict: NSDictionary) {
           let _defaultValue = "N/A"
           self.value = dict["bar"] as? String! ?? _defaultValue
       }
    }

声明_defaultValue并在构造函数中对其进行初始化使代码得以编译。

如果有人可以解释为什么在第一种情况下会发生错误,而在第二种情况下编译器会更快乐,那将是巨大的帮助?

swift swift2
1个回答
14
投票

错误消息的原因是,nil-coalescing运算符定义为

public func ??<T>(optional: T?, defaultValue: @autoclosure () throws -> T) rethrows -> T

并且对第二个参数执行“自动关闭”(以获取短路行为)。所以

self.value = dict["bar"] as? String ?? _defaultValue

由编译器转换为

self.value = dict["bar"] as? String ?? { self._defaultValue }()

这里编译器抱怨,因为self是在之前捕获的正在完全初始化。 (错误消息略有不同在Swift 2和Swift 3之间)。

可能的解决方法。您可以先将属性分配给局部变量:

init(dict: NSDictionary){
    let defValue = _defaultValue
    self.value = dict["bar"] as? String! ?? defValue
}

或者您可以将其设为该类的静态属性:

class Foo {
    static let _defaultValue = "N/A"
    let value: String

    init(dict: NSDictionary) {
        self.value = dict["bar"] as? String ?? Foo._defaultValue
    }
}

或用if语句替换??

class Foo {
    let _defaultValue = "N/A"
    let value: String

    init (dict: NSDictionary) {
        if let value = dict["bar"] as? String {
            self.value = value
        } else {
            self.value = _defaultValue
        }
    }
}

附录:相关资源:

错误报告中的引用:

Jordan Rose:这是正确的,因为&&是使用@autoclosure实现的,但肯定不是最佳选择。

© www.soinside.com 2019 - 2024. All rights reserved.