Swift-继承

  • 一个类可以继承另一个类的方法,属性和其它特性。
  • 当一个类继承其它类时,继承类叫子类,被继承类叫超类(或父类)。
  • 在 Swift 中,继承是区分「类」与其它类型的一个基本特征。
  • 在 Swift 中,类可以调用和访问超类的方法、属性和下标,并且可以重写这些方法,属性和下标来优化或修改它们的行为。
  • Swift 会检查你的重写定义在超类中是否有匹配的定义,以此确保你的重写行为是正确的。
  • 可以为类中继承来的属性添加属性观察器,这样一来,当属性值改变时,类就会被通知到。
  • 可以为任何属性添加属性观察器,无论它原本被定义为存储型属性还是计算型属性。

1. 定义一个基类

  • 不继承于其它类的类,称之为基类。
  • Swift 中的类并不是从一个通用的基类继承而来的。如果你不为自己定义的类指定一个超类的话,这个类就会自动成为基类。
class Vehicle {
    var currentSpeed = 0.0
    var description: String {
        return "traveling at \(currentSpeed) miles per hour"
    }
    func makeNoise() {
        // 什么也不做——因为车辆不一定会有噪音
    }
}

let someVehicle = Vehicle()
print("Vehicle: \(someVehicle.description)")    //  打印“Vehicle: traveling at 0.0 miles per hour”

2. 子类生成

  • 子类生成指的是在一个已有类的基础上创建一个新的类。
  • 子类继承父类的特性,并且可以进一步完善。你还可以为子类添加新的特性。
  • 为了指明某个类的父类,将父类名写在子类名的后面,用冒号分隔:
class SomeClass: SomeSuperclass {
    // 这里是子类的定义
}
class Bicycle: Vehicle {
    var hasBasket = false
}

let bicycle = Bicycle()
bicycle.hasBasket = true

bicycle.currentSpeed = 15.0
print("Bicycle: \(bicycle.description)")    // 打印“Bicycle: traveling at 15.0 miles per hour”
  • 子类还可以继续被其它类继承:
class Tandem: Bicycle {
    var currentNumberOfPassengers = 0
}

let tandem = Tandem()
tandem.hasBasket = true
tandem.currentNumberOfPassengers = 2
tandem.currentSpeed = 22.0
print("Tandem: \(tandem.description)")  // 打印:“Tandem: traveling at 22.0 miles per hour”

3. 重写

  • 子类可以为继承来的实例方法,类方法,实例属性,类属性,或下标提供自己定制的实现,我们把这种行为叫重写。
  • 如果要重写某个特性,你需要在重写定义的前面加上 override 关键字。
  • 任何缺少 override 关键字的重写都会在编译时被认定为错误。
  • override 关键字会提醒 Swift 编译器去检查该类的父类是否有匹配重写版本的声明,这个检查可以确保你的重写定义是正确的。

3.1 访问超类的方法,属性及下标

  • 可以通过使用 super 前缀来访问超类版本的方法,属性或下标
    • 在方法 someMethod() 的重写实现中,可以通过 super.someMethod() 来调用超类版本的 someMethod() 方法。
    • 在属性 someProperty 的 getter 或 setter 的重写实现中,可以通过 super.someProperty 来访问超类版本的 someProperty 属性。
    • 在下标的重写实现中,可以通过 super[someIndex] 来访问超类版本中的相同下标。

3.2 重写方法

  • 在子类中,你可以重写继承来的实例方法或类方法,提供一个定制或替代的方法实现。
class Train: Vehicle {
    override func makeNoise() {
        print("Choo Choo")
    }
}

let train = Train()
train.makeNoise()   // 打印“Choo Choo”

3.3 重写属性

  • 可以重写继承来的实例属性或类型属性,提供自己定制的 getter 和 setter。
  • 或添加属性观察器,使重写的属性可以观察到底层的属性值什么时候发生改变。

3.3.1 重写属性的 Getters 和 Setters

  • 可以提供定制的 getter(或 setter)来重写任何一个继承来的属性,无论这个属性是存储型还是计算型属性。
  • 在重写一个属性时,必须将它的名字和类型都写出来。这样才能使编译器去检查你重写的属性是与超类中同名同类型的属性相匹配的。
  • 可以将一个继承来的只读属性重写为一个读写属性,只需要在重写版本的属性里提供 getter 和 setter 即可。
  • 不可以将一个继承来的读写属性重写为一个只读属性。
class Car: Vehicle {
    var gear = 1
    override var description: String {
        return super.description + " in gear \(gear)"
    }
}

let car = Car()
car.currentSpeed = 25.0
car.gear = 3
print("Car: \(car.description)")    // 打印“Car: traveling at 25.0 miles per hour in gear 3”

重写的 description 属性首先要调用 super.description 返回 Vehicle 类的 description 属性。
之后,Car 类版本的 description 在末尾增加了一些额外的文本来提供关于当前档位的信息。

3.3.2 重写属性观察器

  • 可以通过重写属性为一个继承来的属性添加属性观察器。(无论被继承属性原本是如何实现的,当其属性值发生改变时,你就会被通知到。)
class AutomaticCar: Car {
    override var currentSpeed: Double {
        didSet {
            gear = Int(currentSpeed / 10.0) + 1
        }
    }
}

let automatic = AutomaticCar()
automatic.currentSpeed = 35.0
print("AutomaticCar: \(automatic.description)")
// 打印“AutomaticCar: traveling at 35.0 miles per hour in gear 4”

当你设置 AutomaticCar 的 currentSpeed 属性,属性的 didSet 观察器就会自动地设置 gear 属性,为新的速度选择一个合适的档位。
具体来说就是,属性观察器将新的速度值除以 10,然后向下取得最接近的整数值,最后加 1 来得到档位 gear 的值。例如,速度为 35.0 时,档位为 4。

注意:
    1. 你不可以为继承来的常量存储型属性或继承来的只读计算型属性添加属性观察器。
    2. 这些属性的值是不可以被设置的,所以,为它们提供 willSet 或 didSet 实现也是不恰当。
    3. 此外还要注意,你不可以同时提供重写的 setter 和重写的属性观察器。如果你想观察属性值的变化,并且你已经为那个属性提供了定制的 setter,那么你在 setter 中就可以观察到任何值变化了。

4. 防止重写

  • 可以通过把方法,属性或下标标记为 final 来防止它们被重写。
  • 试图对带有 final 标记的方法、属性或下标进行重写的代码,都会在编译时会报错。
  • 在类扩展中的方法,属性或下标也可以在扩展的定义里标记为 final。
  • 可以通过在关键字 class 前添加 final 修饰符(final class)来将整个类标记为 final 。这样的类是不可被继承的,试图继承这样的类会导致编译报错。

你可能感兴趣的:(Swift-继承)