【问题标题】:confusion over initialisation in swift快速初始化的困惑
【发布时间】:2016-06-30 22:59:06
【问题描述】:

根据“Swift 编程语言 2.1”,如果子类为其所有属性提供默认值,则子类自动继承其所有超类的指定初始化器和便利初始化器。所以对于下面的代码,类ShoppingListItem应该从它的超类RecipeIngredient继承三个初始化器,即

//1
convenience init() {
    self.init(name: "[unnamed]")
}

//2
convenience init(name: String){
    self.init(name: name, quantity: 1)
}

//3
init(name: String, quantity: Int){
    self.quantity = quantity
    super.init(name: name)
}

如果我想创建一个 ShoppingListItem 的实例,我会写

let ingredientThree = ShoppingListItem(name: "apple", quantity: 10)

这将调用init(name: String, quantity: Int){},但在这个初始化程序中,有一个对super.init(name: name) 的调用,这就是我卡住的地方。当初始化过程达到super.init(name: name) 时会发生什么?它是否调用ShoppingListItem 的超类init(name: String){} 初始化程序?还是没有?这里到底发生了什么?有人可以向我解释一下吗?提前感谢您的帮助!

实际代码:

class Food {
    var name: String
    init(name: String){
        self.name = name
    }
    convenience init() {
        self.init(name: "[unnamed]")
    }

}

class RecipeIngredient: Food{
    var quantity: Int
    init(name: String, quantity: Int){
        self.quantity = quantity
        super.init(name: name)
    }

    override convenience init(name: String){
        self.init(name: name, quantity: 1)
    }
}

class ShoppingListItem: RecipeIngredient {
    var purchased = false
    var description: String {
        var output = "\(quantity) X \(name)"
        output += purchased ? " ✔" : " ✘"
        return output
    }
}

来自 Swift 编程语言 2.1 的图表

【问题讨论】:

  • 有时当我看到来自 Apple 文档的示例代码时,我想去库比蒂诺并给他们的作者一巴掌:ShoppingListItem 不是一种成分——它有一种成分。它应该被建模为组合,而不是继承。一种成分有一种食物:又是成分。

标签: ios swift initialization subclass superclass


【解决方案1】:

我将您的代码放入游乐场并添加了一些打印件。这是发生了什么:

  • RecipeIngredient.init(String, Int)
  • Food.init

顺便说一句,您问过它是否会调用 ShoppingListItem 的超类的 init(name: String){} 初始化程序。

答案是否定的。你用过的

let ingredientThree = RecipeIngredient(name: "apple", quantity: 10)

这与 ShoppingListItem 无关。

一般来说,如果你有类继承 Class1

class Food {
    var name: String
    init(name: String){
        print("Food.init")
        self.name = name
    }
    convenience init() {
        self.init(name: "[unnamed]")
    }

}

class RecipeIngredient: Food{
    var quantity: Int
    init(name: String, quantity: Int){
        print("RecipeIngredient.init(String, Int)")
        self.quantity = quantity
        super.init(name: name)
    }

    override convenience init(name: String){
        self.init(name: name, quantity: 1)
    }
}

class ShoppingListItem: RecipeIngredient {
    var purchased = false
    var description: String {
        var output = "\(quantity) X \(name)"
        output += purchased ? " ✔" : " ✘"
        return output
    }
}
let ingredientThree = RecipeIngredient(name: "apple", quantity: 10)

【讨论】:

  • 对不起,我的真正意思是让成分三= ShoppingListItem(名称:“苹果”,数量:10),而不是让成分三=食谱成分(名称:“苹果”,数量:10)。您能否重新调整此调整的答案?对你的帮助表示感谢!!上帝保佑你!!
  • 我假设您可以将代码放入 Playground 并进行更改。你会看到什么被调用。它的工作原理真的没有什么神奇之处。与Java、PHP、C++等相同。
猜你喜欢
  • 2018-04-25
  • 1970-01-01
  • 2021-10-26
  • 2019-02-17
  • 1970-01-01
  • 2023-03-08
  • 2016-05-27
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多