是什么导致“常量在初始化之前被闭包捕获”错误

在下一堂课

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在构造函数中声明和初始化它可以使代码编译。

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

回答:

该错误信息的原因是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

}

}

}

附录: 相关资源:

  • 在Swift论坛中的初始化程序中,自我自动闭合。
  • SR-944在初始化所有属性的错误报告之前,不能在&&的RHS中使用“自我”。

引用错误报告:

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

以上是 是什么导致“常量在初始化之前被闭包捕获”错误 的全部内容, 来源链接: utcz.com/qa/414076.html

回到顶部