swift - 是什么导致 'Constant captured by a closure before being initialized'错误

标签 swift swift2

在接下来的类(class)中

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

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

编译器失败并显示消息 在初始化之前由闭包捕获的常量“self.value”

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

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

class Foo {
       let value: String 

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

声明 _defaultValue 并在构造函数中初始化它使代码可以编译。

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

最佳答案

错误消息的原因是 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: This is true since && is implemented using @autoclosure, but it's certainly suboptimal.

关于swift - 是什么导致 'Constant captured by a closure before being initialized'错误,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54390382/

相关文章:

ios - Storyboard的 Swift 输入和事件处理

macos - Swift2:初始化 UnsafeMutablePointer<Unmanaged<CFMutableDictionary>?> 参数以传递给 IORegistryEntryCreateCFProperties 的正确方法

string - 从常量 Ints 创建 Range<String.Index>

ios - 单击按钮时无法播放一系列图像

ios - prepareForSegue 不能快速工作。我该如何解决?

ios - UITableViewAutomaticDimension 在滚动之前不工作

ios - 如何指定 uicollectionView 背景 View 的最小尺寸?

swift - 如何在 spritekit 中创建垂直滚动菜单?

swift - Swift 的 catch block 前面的 _ 下划线是什么?

xcode - swift 2.0 : Type of expression is ambiguous without more context (Using Parse)