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

标签 swift function class closures

问题是以下编译器错误:

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

相关文章:

c++:如何在不创建成员函数的情况下声明 std::functions 以分配给它们

c++ - 在 Objective C 中添加 "->"

python - 在 Python 类中继承 Cython 类

ios - 使用自定义 uitableViewCell 在 tableView 上按下回车键或单击时隐藏键盘

c - 我的函数一直返回 0 而不是实际计算的返回值?

function - 使用回调函数传递附加参数

java - 如何通过从第一个类调用第二个类的方法来调用第三个类的方法

ios - 我应该使用 ImageView 还是 TextField?

swift - 你如何获得 couchbase 数据库的大小

ios - 使用 Google Firebase 同时上传多张图片