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

时间:2018-08-17 03:36:11

标签: 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定义上出现编译器错误。 据我所知,闭包和函数基本上是相同的。那么,它们之间有什么区别使上述闭包受到限制?

1 个答案:

答案 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。