问题是以下编译器错误:
“实例成员‘name’不能用于类型‘Person’”
class Person {
var name: String = "Amir"
var myClosure = { (family: String) -> Void in
print(name + " " + family)
}
func myFunc(family: String) -> Void {
print(name + " " + family)
}
}
myFunc 和 myClosure 中的代码是相同的,但 myClosure 定义有编译器错误。 据我所知,闭包和功能基本上是相同的。那么它们之间的区别是什么导致了上述对于闭包的限制呢?
最佳答案
问题是我正在尝试使用实例的变量来初始化变量,而实例本身尚未创建。 所以这个问题与闭包性质无关。以下代码中确实存在同样的问题,我尝试使用名称实例属性初始化昵称。
class Person {
var name: String = "Amirreza"
var nickname: String = name
}
正如编译器所说,“属性初始值设定项在‘self’可用之前运行”,因此在上面的代码中,self.name 无法用于初始化昵称,这就是问题的根源。
这里的解决方案是在昵称定义之前使用“lazy”关键字。惰性键盘将昵称属性的初始化推迟到第一次访问时。意味着当我们确定实例已创建且 name 属性可用时。
要了解有关惰性的更多信息,请参阅https://docs.swift.org/swift-book/LanguageGuide/Properties.html
所以上述代码的正确形式是:
class Person {
var name: String = "Amirreza"
lazy var nickname: String = name
}
我的主要问题代码的正确形式是:
class Person {
var name: String = "Amirreza"
lazy var myClosure = { [weak self] (family: String) -> Void in
print(self!.name + " " + family)
}
func myFunc(family: String) -> Void {
print(self.name + " " + family)
}
}
请注意,我们必须使用 self 显式引用闭包内的属性,如 self.name。同时,为了避免强引用循环,我们需要在闭包定义中定义一个捕获列表,在我的示例中是[weak self]。 通过将 self 定义为“weak”,self 就变成了一个可选变量,所以我们必须以某种方式解开它,这使得我在闭包中写入 self!.name 。
关于swift - 为什么我们不能在闭包定义中使用实例成员?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51888048/