Swift learning part 13 - 继承

一个类可以继承另一个类的方法、属性和其它特性。当一个类继承其它类时,继承类叫子类,被继承类叫超类(或父类)。在 Swift 中,继承是区分 类 与其它类型的一个基本特征。

在 Swift 中,类可以调用和访问超类的方法、属性和下标,并且可以重写这些方法、属性和下标来修改它们的行为。Swift 会检查你的重写定义在超类中是否有匹配的定义,以此确保你的重写行为是正确的。

可以为类中继承来的属性添加属性观察器,这样一来,当属性值改变时,类就会被通知到。可以为任何属性添加属性观察器,无论它原本被定义为存储型属性还是计算型属性。

定义一个基类

不继承于其它类的类,称之为基类。Swift 中的类并不是从一个通用的基类继承而来,如果不为定义的类指定一个超类,这个类就自动成为基类。

下面的例子定义了一个叫 Vehicle 的基类。这个基类声明了一个名为 currentSpeed,默认值是 0.0 的存储属性(属性类型推断为Double)。currentSpeed 属性的值被一个 String 类型的只读计算型属性 description 使用,用来创建车辆的描述。

Vehicle基类 定义了一个名为 makeNoise 的方法。这个方法实际上不为 Vehicle 实例做任何事,但之后将会被 Vehicle 的子类定制:

class Vehicle {
    var currentSpeed = 0.0
    var description: String {
        return "traveling at \(currentSpeed) miles per hour"
    }
    
    func makeNoise() {
        // 由子类继承并实现
    }
}

初始化 Vehicle 实例,并访问它的 description 属性:

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

子类生成

子类生成指的是在一个已有类的基础上创建一个新的类。子类继承超类的特性,并且可以进一步完善,子类还可以添加新的特性。

为了指明某个类的超类,将超类名写在子类名的后面,用冒号分离。下面例子定义了一个叫 Bicycle 的子类,继承父类 Vehicle:

class Bicycle: Vehicle {
    var hasBasket = false
}

新的 Bicycle 类自动获得 Vehicle 类的所有特性,比如 currentSpeed 和 description 属性,还有 makeNoise() 方法。此外,Bicycle 类还定义了一个默认值为 false 的存储属性 hasBasket。

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”

重写

子类可以为继承来的实例方法,类方法,实例属性或下标提供自己定制的实现,我们把这种行为叫重写。

如果要重写某个特性,你需要在重写定义的前面加上override关键字。这么做,你就表明了你是想提供一个重写版本,而非错误地提供了一个相同的定义。意外的重写行为可能会导致不可预知的错误,任何缺少override关键字的重写都会在编译时被诊断为错误。

override关键字会提醒 Swift 编译器去检查该类的超类(或其中一个父类)是否有匹配重写版本的声明。这个检查可以确保你的重写定义是正确的。

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

当你在子类中重写超类的方法、属性或下标时,有时在你的重写版本中使用已经存在的超类实现会大有裨益。比如,你可以完善已有实现的行为,或在一个继承来的变量中存储一个修改过的值。

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

重写方法

在子类中,可以重写继承来的实例方法或类方法。

下面的例子定义了 Vehicle 的一个新的子类,叫 Train,它重写了从 Vehicle 类继承来的 makeNoise() 方法:

class Train: Vehicle {
    override func makeNoise() {
        print("Choo Choo")
    }
}

创建 Train 实例并调用 makeNoise() 方法:

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

重写属性

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

重写属性的 Getters 和 Setters

可以提供定制的 getter(或 setter)来重写任意继承来的属性,无论继承来的属性是存储型的还是计算型的属性。子类并不知道继承来的属性是存储型的还是计算型的,它只知道继承来的属性会有一个名字和类型。在重写一个属性时,必需将它的名字和类型都写出来。这样才能使编译器去检查你重写的属性是与超类中同名同类型的属性相匹配的。

你可以将一个继承来的只读属性重写为一个读写属性,只需要在重写版本的属性里提供 getter 和 setter 即可。但是,你不可以将一个继承来的读写属性重写为一个只读属性

注意:
如果你在重写属性中提供了 setter,那么你也一定要提供 getter。如果你不想在重写版本中的 getter 里修改继承来的属性值,你可以直接通过super.someProperty来返回继承来的值,其中someProperty是你要重写的属性的名字。

以下的例子定义了一个新类,叫Car,它是Vehicle的子类。Car 类重写了继承自 Vehicle 的 description 属性,提供包含当前档位的自定义描述:

class Car: Vehicle {
    var gear = 1
    override var description: String {
        return super.description + "in gear \(gear)"
    }
}

创建 Car 实例并设置它的 gear 和 currentSpeed 属性,可以看到它的 description 返回了 Car 中的自定义描述:

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”

重写属性观察器

可以通过重写属性为一个继承来的属性添加属性观察器。无论被继承的属性原本是如何实现的,当其属性值发生改变时,你就会被通知到。

注意:
你不可以为继承来的常量存储型属性或继承来的只读计算型属性添加属性观察器。这些属性的值是不可以被设置的,所以,为它们提供 willSet 或 didSet 实现是不恰当。

此外还要注意,你不可以同时提供重写的 setter 和重写的属性观察器。如果你想观察属性值的变化,并且你已经为那个属性提供了定制的 setter,那么你在 setter 中就可以观察到任何值变化了。

下面的例子定义了一个新类叫 AutomaticCar,它是Car的子类。AutomaticCar 表示自动挡汽车,它可以根据当前的速度自动选择合适的挡位。

class AutomaticCar: Car {
    override var currentSpeed: Double {
        didSet {
            gear = Int(currentSpeed / 10.0) + 1
        }
    }
}
let automaticCar = AutomaticCar()
automaticCar.currentSpeed = 35.0 // 速度为35.0时,档位为4
print("AutomaticCar: \(automaticCar.description)")
// 打印“AutomaticCar: traveling at 35.0 miles per hour in gear 4”

防止重写

你可以通过把方法、属性或下标标记为 final 来防止它们被重写,只需要在声明关键字前加上final修饰符即可(例如:final var,final func,final class func,以及final subscript)。

如果你重写了带有 final 标记的方法,属性或下标,在编译时会报错。在类扩展中的方法,属性或下标也可以在扩展的定义里标记为 final。

你可以通过在关键字 class 前添加 final 修饰符(final class)来将整个类标记为 final。这样的类是不可被继承的,试图继承这样的类会导致编译报错。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值