问题是以下编译器错误:
“实例成员'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定义上出现编译器错误。 据我所知,闭包和函数基本上是相同的。那么,它们之间有什么区别使上述闭包受到限制?
答案 0 :(得分:0)
问题是我正在尝试使用实例的变量初始化变量,而实例本身尚未创建。 因此,问题与关闭性质无关。下面的代码中确实存在相同的问题,我尝试使用名称实例属性初始化昵称。
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之类的属性。同时,为了避免强引用循环,我们需要在闭包定义中定义一个捕获列表,在我的示例中,该列表为[弱自我]。 通过将self定义为“弱”,self变成一个可选变量,因此我们必须以某种方式对其进行拆包,这使我不得不在闭包中编写self!.name。