我有两门课,形状和正方形
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 ?
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中并非如此。在执行其他操作之前,包括调用方法(包括父类的初始化式),您有责任确保实例变量处于一致的状态。
在初始化所有实例变量后,应该调用"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
}
}
对不起,格式很难看。
只要在声明后加上一个问题字符,一切都会好起来的。
一个问题告诉编译器这个值是可选的。
class Square: Shape {
var sideLength: Double? // <=== like this ..
init(sideLength:Double, name:String) {
super.init(name:name) // Error here
self.sideLength = sideLength
numberOfSides = 4
}
func area () -> Double {
return sideLength * sideLength
}
}
Edit1:
有一种更好的方法可以跳过这个错误。根据jmaschad的评论,没有理由在你的情况下使用可选的,因为可选的是不舒服的使用,你总是要检查如果可选的不是nil之前访问它。所以你所要做的就是在声明后初始化member:
class Square: Shape {
var sideLength: Double=Double()
init(sideLength:Double, name:String) {
super.init(name:name)
self.sideLength = sideLength
numberOfSides = 4
}
func area () -> Double {
return sideLength * sideLength
}
}
Edit2:
在这个答案上得到两个负数后,我找到了更好的方法。如果你想在构造函数中初始化类成员,你必须在构造函数中并在super.init()调用之前给它赋初始值。是这样的:
class Square: Shape {
var sideLength: Double
init(sideLength:Double, name:String) {
self.sideLength = sideLength // <= before super.init call..
super.init(name:name)
numberOfSides = 4
}
func area () -> Double {
return sideLength * sideLength
}
}
祝你学习Swift好运。