我有两门课,形状和正方形

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 ?


当前回答

对不起,格式很难看。 只要在声明后加上一个问题字符,一切都会好起来的。 一个问题告诉编译器这个值是可选的。

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好运。

其他回答

爱德华,

你可以像这样修改你的例子中的代码:

var playerShip:PlayerShip!
var deltaPoint = CGPointZero

init(size: CGSize)
{
    super.init(size: size)
    playerLayerNode.addChild(playerShip)        
}

这使用了一个隐式解包装的可选选项。

在文档中我们可以读到:

与可选选项一样,如果你不提供初始值 声明一个隐式展开的可选变量或属性,它是 Value自动默认为nil。”

Swift强制你在使用/可能使用之前初始化每个成员var。因为它不能确定当它变成超级时发生了什么,所以它会出错:小心总比后悔好

@Janos如果你让属性是可选的,你不需要在init中初始化它。- - - - - -

这对我很管用。

Swift不允许你在不初始化属性的情况下初始化超类,与Obj c相反,所以你必须在调用"super.init"之前初始化所有属性。

请登录http://blog.scottlogic.com/2014/11/20/swift-initialisation.html。 它很好地解释了你的问题。

对不起,格式很难看。 只要在声明后加上一个问题字符,一切都会好起来的。 一个问题告诉编译器这个值是可选的。

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好运。