微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

在swift中调用超级指定初始化程序之前初始化属性值的目的是什么?

这是我的代码

class Base
{
    init(){
        print("Super!")
    }
}

class Test : Base
{
    internal var y:Int
    convenience init(_ a:Int)
    {
        self.init()
        print("\(a)")
    }
    override init()
    {
        super.init() //Error!!! Property 'self.y' not initialized at super.init call
        y = 123
    }
}

我认为应该编译:

y在类’Base’中是不可见的,y和超类的初始化顺序是否真的无关紧要.

解决方法

你的论点

I think this should be compiled:

y is not visible inside class ‘Base’,whether order of initializations
of y’s and super class’s doesn’t really matter.

这是不正确的,这是不安全的.

超类init可以调用实例
在子类中重写的方法.那是(至少一个)
之所以必须在调用super.init()之前初始化所有子类属性的原因.

一个简单的例子:

class Base
{
    init(){
        print("enter Base.init")
        setup()
        print("leave Base.init")
    }

    func setup() {
        print("Base.setup called")
    }
}

class Test : Base
{
    internal var y:Int
    override init()
    {
        y = 123
        print("before super.init")
        super.init()
        print("after super.init")
    }

    override func setup() {
        print("Test.setup called")
        print("y = \(y)")
    }
}

输出

before super.init
enter Base.init
Test.setup called
y = 123
leave Base.init
after super.init

如您所见,访问子类的y属性
在super.init()调用期间,即使它不知道
超类.

比较Objective-C中的情况可能会很有趣
其中self = [super initXXX]总是先调用.这有
结果是属性访问init / dealloc方法中的self.prop
是不安全的,直接访问实例变量_prop是
建议因为对象可能处于“部分构造状态”.
参见例如Should I refer to self.property in the init method with ARC?.

所以这是Swift已经解决的问题之一(以更严格的要求为代价).

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 [email protected] 举报,一经查实,本站将立刻删除。

相关推荐