Swift—默认构造函数-备

时间:2021-12-30 11:12:43

结构体和类的实例在构造过程中会调用一种特殊的init方法,称为构造函数。构造函数没有返回值,可以重载。在多个构造函数重载的情况下,运行环境可以根据它的外部参数名或参数列表调用合适的构造函数。
默认构造函数
结构体和类在构造过程中会调用一个构造函数,即便是没有编写任何构造函数,编译器也会提供一个默认的构造函数。下面看示例代码:

  1. class Rectangle {
  2. var width: Double  = 0.0
  3. var height: Double = 0.0
  4. }
  5. var rect = Rectangle()   //创建实例,并调用默认构造函数init()
  6. rect.width = 320.0
  7. rect.height = 480.0
  8. print("长方形:\(rect.width) x \(rect.height)")
  9. Rectangle()表示调用了某个方法,这个方法就是默认构造函数init()。
  10. 事实上,在Rectangle的定义过程中省略了构造函数,相当于如下代码:
  11. class Rectangle {
  12. var width: Double  = 0.0
  13. var height: Double = 0.0
  14. init() {
  15. }
  16. }

如果Rectangle是结构体,则它的定义如下:
struct Rectangle {
    var width: Double = 0.0
    var height: Double = 0.0
}
而结构体Rectangle的默认构造函数与类Rectangle的默认构造函数是不同的,相当于如下代码:

  1. struct Rectangle {
  2. var width: Double = 0.0
  3. var height: Double = 0.0
  4. init() {
  5. }
  6. init(width: Double, height: Double) { //有参数的构造函数
  7. self.width   = width
  8. self.height  = height
  9. }
  10. }

要调用哪个构造函数是根据传递的参数名和参数类型决定的。

在构造函数中可以使用构造函数代理帮助完成部分构造工作。类构造函数代理分为横向代理和向上代理,横向代理只能在发生在同一类内部,这种构造函数称为便利构造函数。向上代理发生在继承的情况下,在子类构造过程中,要先调用父类构造函数初始化父类的存储属性,这种构造函数称为指定构造函数。

构造函数调用规则

  1. Person和Student类示例:
  2. class Person {
  3. var name: String
  4. var age: Int
  5. func description() -> String {
  6. return "\(name) 年龄是: \(age)"
  7. }
  8. convenience init () {         //便利构造函数
  9. self.init(name: "Tony")
  10. self.age = 18
  11. }
  12. convenience init (name: String) { //便利构造函数
  13. self.init(name: name, age: 18)
  14. }
  15. init (name: String, age: Int) {       //指定构造函数
  16. self.name = name
  17. self.age  = age
  18. }
  19. }
  20. class Student: Person {
  21. var school: String
  22. init (name: String, age: Int, school: String) {       //指定构造函数
  23. self.school = school
  24. super.init(name: name, age: age)
  25. }
  26. convenience override init (name: String, age: Int) {//便利构造函数
  27. self.init(name: name, age: age, school: "清华大学")
  28. }
  29. }
  30. let student = Student()
  31. print("学生: \(student.description())")

构造函数之间的调用形成了构造函数链,如图所示。

Swift限制构造函数之间的代理调用的规则有3条,如下所示。

指定构造函数必须调用其直接父类的的指定构造函数。从图可见,Student中的④号指定构造函数调用Person中的③号指定构造函数。

    • 便利构造函数必须调用同一类中定义的其他构造函数。从图可见,Student中的⑤号便利构造函数调用同一类中的④号便利构造函数,Person中的①号便利构造函数调用同一类中的②号便利构造函数。

    • 便利构造函数必须最终以调用一个指定构造函数结束。从图可见,Student中的⑤号便利构造函数调用同一类中的④号指定构造函数,Person中的②号便利构造函数调用同一类中的③号指定构造函数。

  • ==================================================
  • Swift中的子类构造函数的来源有两种:自己编写和从父类继承。并不是父类的所有的构造函数都能继承下来,能够从父类继承下来的构造函数是有条件的,如下所示。

    • 条件1:如果子类没有定义任何指定构造函数,它将自动继承所有父类的指定构造函数。

    • 条件2:如果子类提供了所有父类指定构造函数的实现,无论是通过条件1继承过来的,还是通过自己编写实现的,它都将自动继承所有父类的便利构造函数。

    下面看示例代码:

    1. class Person {
    2. var name: String
    3. var age: Int
    4. func description() -> String {
    5. return "\(name) 年龄是: \(age)"
    6. }
    7. convenience init () {
    8. self.init(name: "Tony")
    9. self.age = 18
    10. }
    11. convenience init (name: String) {
    12. self.init(name: name, age: 18)
    13. }
    14. init (name: String, age: Int) {
    15. self.name = name
    16. self.age  = age
    17. }
    18. }
    19. class Student: Person {
    20. var school: String
    21. init (name: String, age: Int, school: String) {
    22. self.school = school
    23. super.init(name: name, age: age)
    24. }
    25. convenience override init (name: String, age: Int) {
    26. self.init(name: name, age: age, school: "清华大学")
    27. }
    28. }
    29. class Graduate: Student {
    30. var special: String = ""
    31. }

    来看看符合条件1的继承,Graduate继承Student,Graduate类没有定义任何指定构造函数,它将自动继承所有Student的指定构造函数。符合条件1后,Graduate从Student继承了如下指定构造函数:

    init (name: String, age: Int,school: String)

    再看符合条件2的继承,由于Graduate实现了Student的所有指定构造函数,Graduate将自动继承所有Student的便利构造函数。符合条件2后,Graduate从Student继承了如下3个便利构造函数:

    init (name: String, age: Int)

    init (name: String)

    init ()

    Student继承Person后有4个构造函数。

    条件1对Student不满足,因为它有指定构造函数,Student类中的便利构造函数init (name: String, age: Int)满足了条件2,它实现了父类指定构造函数init (name: String, age: Int)。另外,由于子类构造函数与父类构造函数参数相同,需要使用override关键字,表示子类构造函数重写(overriding)了父类构造函数。

    由于Student类实现了父类指定构造函数,因此也继承了父类的另外两个便利构造函数。