gpt4 book ai didi

swift - 为什么我们不能在闭包定义中使用实例成员?

转载 作者:行者123 更新时间:2023-11-30 11:14:21 24 4
gpt4 key购买 nike

问题是以下编译器错误:

“实例成员‘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/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com