我有两门课,形状和正方形
class Shape {
var numberOfSides = 0
var name: String
init(name:String) {
self.name = name
}
func simpleDescription() -> String {
return "A shape with \(numberOfSides) sides."
}
}
class Square: Shape {
var sideLength: Double
init(sideLength:Double, name:String) {
super.init(name:name) // Error here
self.sideLength = sideLength
numberOfSides = 4
}
func area () -> Double {
return sideLength * sideLength
}
}
通过上面的实现,我得到了错误:
property 'self.sideLength' not initialized at super.init call
super.init(name:name)
为什么我要设置self。调用super.init之前的sidelth ?
在初始化所有实例变量后,应该调用"super.init()"。
在Apple的“Intermediate Swift”视频中(你可以在Apple Developer视频资源页面https://developer.apple.com/videos/wwdc/2014/中找到它),大约28:40,它明确地说,在初始化实例变量后,必须调用超类中的所有初始化器。
在Objective-C中,情况正好相反。在Swift中,由于所有属性在使用之前都需要初始化,所以我们需要先初始化属性。这是为了防止从超类的“init()”方法调用覆盖函数,而不首先初始化属性。
所以“Square”的实现应该是:
class Square: Shape {
var sideLength: Double
init(sideLength:Double, name:String) {
self.sideLength = sideLength
numberOfSides = 4
super.init(name:name) // Correct position for "super.init()"
}
func area () -> Double {
return sideLength * sideLength
}
}
Swift在初始化器中有一个非常清晰、特定的操作序列。让我们从一些基本的例子开始,一直到一般的情况。
让我们以一个对象a为例,我们将这样定义它。
class A {
var x: Int
init(x: Int) {
self.x = x
}
}
注意,A没有超类,所以它不能调用super.init()函数,因为它不存在。
好,现在我们用一个新类B来子类A。
class B: A {
var y: Int
init(x: Int, y: Int) {
self.y = y
super.init(x: x)
}
}
这与Objective-C不同,Objective-C通常首先调用[super init]。但在Swift中并非如此。在执行其他操作之前,包括调用方法(包括父类的初始化式),您有责任确保实例变量处于一致的状态。