在 Swift 中,继承是区分「类」与其它类型的一个基本特征。swift不支持多重继承。类能够调用和访问超类的方法,属性和subscripts下标,而且能够重写(override)这些方法,属性和附属脚原本优化或修改它们的行为。Swift 会检查你的重写定义在超类中是否有匹配的定义,以此确保你的重写行为是正确的。swift
能够为类中继承来的属性添加属性观察器(property observer),这样一来,当属性值改变时,类就会被通知到。能够为任何属性添加属性观察器,不管它本来被定义为存储型属性(stored property)仍是计算型属性(computed property)。
基类的定义ide
不继承于其它类的类,称之为基类(base calss)函数
注意:Swift 中的类并非从一个通用的基类继承而来。若是你不为你定义的类指定一个超类的话,这个类就自动成为基类。
下面的例子定义了一个叫Vehicle的基类。这个基类声明了两个对全部车辆都通用的属性(numberOfWheels和maxPassengers)。这些属性在description方法中使用,这个方法返回一个String类型的,对车辆特征的描述:
class Vehicle { var numberOfWheels: Int var maxPassengers: Int func description() -> String { return "\(numberOfWheels) wheels; up to \(maxPassengers) passengers" } init() { numberOfWheels = 0 maxPassengers = 1 } }
子类化
子类化是子类继承超类的特性,而且能够优化或改变它。你还能够为子类添加新的特性。
为了指明某个类的超类,将超类名写在子类名的后面,用冒号分隔:
class SomeClass: SomeSuperclass { // 类的定义
}
下一个例子,定义一个更具体的车辆类叫Bicycle。这个新类是在 Vehicle类的基础上建立起来。所以你须要将Vehicle类放在 Bicycle类后面,用冒号分隔。
class Bicycle: Vehicle { init() { super.init() numberOfWheels = 2 } }
Bicycle是Vehicle的子类,Vehicle是Bicycle的超类。新的Bicycle类自动得到Vehicle类的特性,好比 maxPassengers和numberOfWheels属性。你能够在子类中定制这些特性,或添加新的特性来更好地描述Bicycle类。
Bicycle类定义了一个构造器来设置它定制的特性(自行车只有2个轮子)。Bicycle的构造器调用了它父类Vehicle的构造器 super.init(),以此确保在Bicycle类试图修改那些继承来的属性前Vehicle类已经初始化过它们了。
注意:不像 Objective-C,在 Swift 中,初始化器默认是不继承的。
Vehicle类中maxPassengers的默认值对自行车来讲已是正确的,所以在Bicycle的构造器中并无改变它。而numberOfWheels原来的值对自行车来讲是不正确的,所以在初始化器中将它更改成 2。
Bicycle不只能够继承Vehicle的属性,还能够继承它的方法。若是你建立了一个Bicycle类的实例,你就能够调用它继承来的description方法,而且能够看到,它输出的属性值已经发生了变化:
let bicycle = Bicycle() println("Bicycle: \(bicycle.description())") // Bicycle: 2 wheels; up to 1 passengers
子类还能够继续被其它类继承:优化
class Tandem: Bicycle { init() { super.init() maxPassengers = 2 } }
意:子类只容许修改从超类继承来的变量属性,而不能修改继承来的常量属性。spa
重写(Overriding)code
子类能够为继承来的实例方法(instance method),类方法(class method),实例属性(instance property),或subscript提供本身的实现(implementation)。咱们把这种行为叫重写(overriding)。使用override关键字来实现,意外的重写行为可能会致使不可预知的错误,任何缺乏override关键字的重写都会在编译时检查为错误。override关键字会提醒 Swift 编译器去检查该类的超类(或其中一个父类)是否有匹配重写版本的声明。这个检查能够确保你的重写定义是正确的。
访问超类的方法,属性
经过使用super来访问超类的属性、方法、下标,在方法someMethod的重写实现中,能够经过super.someMethod()来调用超类版本的someMethod方法。在属性someProperty的 getter 或 setter 的重写实现中,能够经过super.someProperty来访问超类版本的someProperty属性。在附属脚本的重写实现中,能够经过super[someIndex]来访问超类版本中的相同下标。
1.重写方法
class Car: Vehicle { var speed: Double = 0.0 init() { super.init() maxPassengers = 5 numberOfWheels = 4 } override func description() -> String { return super.description() + "; "
+ "traveling at \(speed) mph" } }
Car声明了一个新的存储型属性speed,它是Double类型的,默认值是0.0,表示“时速是0英里”。Car有本身的初始化器,它将乘客的最大数量设为5,轮子数量设为4。
Car重写了继承来的description方法,它的声明与Vehicle中的description方法一致,声明前面加上了override关键字。
Car中的description方法并不是彻底自定义,而是经过super.description使用了超类Vehicle中的description方法,而后再追加一些额外的信息,好比汽车的当前速度。
若是你建立一个Car的新实例,并打印description方法的输出,你就会发现描述信息已经发生了改变
let car = Car() println("Car: \(car.description())") // Car: 4 wheels; up to 5 passengers; traveling at 0.0 mph
2.
重写属性
你能够重写继承来的实例属性或类属性,提供本身定制的getter和setter,或添加属性观察器使重写的属性观察属性值何时发生改变。
3.重写属性的Getters和Setters
你能够提供定制的 getter(或 setter)来重写任意继承来的属性,不管继承来的属性是存储型的仍是计算型的属性。子类并不知道继承来的属性是存储型的仍是计算型的,它只知道继承来的属性会有一个名字和类型。你在重写一个属性时,必需将它的名字和类型都写出来。这样才能使编译器去检查你重写的属性是与超类中同名同类型的属性相匹配的。
你能够将一个继承来的只读属性重写为一个读写属性,只须要你在重写版本的属性里提供 getter 和 setter 便可。可是,你不能够将一个继承来的读写属性重写为一个只读属性。
注意:若是你在重写属性中提供了 setter,那么你也必定要提供 getter。若是你不想在重写版本中的 getter 里修改继承来的属性值,你能够直接返回super.someProperty来返回继承来的值。正以下面的SpeedLimitedCar的例子所示。
如下的例子定义了一个新类,叫SpeedLimitedCar,它是Car的子类。类SpeedLimitedCar表示安装了限速装置的车,它的最高速度只能达到40mph。你能够经过重写继承来的speed属性来实现这个速度限制
class SpeedLimitedCar: Car { override var speed: Double { get { return super.speed } set { super.speed = min(newValue, 40.0) } } }
当你设置一个SpeedLimitedCar实例的speed属性时,属性setter的实现会去检查新值与限制值40mph的大小,它会将超类的speed设置为newValue和40.0中较小的那个。这两个值哪一个较小由min函数决定,它是Swift标准库中的一个全局函数。min函数接收两个或更多的数,返回其中最小的那个。
若是你尝试将SpeedLimitedCar实例的speed属性设置为一个大于40mph的数,而后打印description函数的输出,你会发现速度被限制在40mph:
let limitedCar = SpeedLimitedCar() limitedCar.speed = 60.0 println("SpeedLimitedCar: \(limitedCar.description())") // SpeedLimitedCar: 4 wheels; up to 5 passengers; traveling at 40.0 mph
4.重写属性观察器(Property Observer)
你能够在属性重写中为一个继承来的属性添加属性观察器。这样一来,当继承来的属性值发生改变时,你就会被通知到,不管那个属性本来是如何实现的。关于属性观察器的更多内容,请看属性观察器。
注意:你不能够为继承来的常量存储型属性或继承来的只读计算型属性添加属性观察器。这些属性的值是不能够被设置的,因此,为它们提供willSet或didSet实现是不恰当。此外还要注意,你不能够同时提供重写的 setter 和重写的属性观察器。若是你想观察属性值的变化,而且你已经为那个属性提供了定制的 setter,那么你在 setter 中就能够观察到任何值变化了。
下面的例子定义了一个新类叫AutomaticCar,它是Car的子类。AutomaticCar表示自动挡汽车,它能够根据当前的速度自动选择合适的挡位。AutomaticCar也提供了定制的description方法,能够输出当前挡位。
class AutomaticCar: Car { var gear = 1
override var speed: Double { didSet { gear = Int(speed / 10.0) + 1 } } override func description() -> String { return super.description() + " in gear \(gear)" } }
当你设置AutomaticCar的speed属性,属性的didSet观察器就会自动地设置gear属性,为新的速度选择一个合适的挡位。具体来讲就是,属性观察器将新的速度值除以10,而后向下取得最接近的整数值,最后加1来获得档位gear的值。例如,速度为10.0时,挡位为1;速度为35.0时,挡位为4:server
let automatic = AutomaticCar() automatic.speed = 35.0 println("AutomaticCar: \(automatic.description())") // AutomaticCar: 4 wheels; up to 5 passengers; traveling at 35.0 mph in gear 4
防止重写blog
你能够经过把方法,属性或下标标记为final来防止它们被重写,只须要在声明关键字前加上final特性便可。(例如:final var, final func, final class func, 以及 final subscript)
若是你重写了final方法,属性或下标,在编译时会报错。在扩展中,你添加到类里的方法,属性或下标也能够在扩展的定义里标记为 final。
你能够经过在关键字class前添加@final特性(final class)来将整个类标记为 final 的,这样的类是不可被继承的,不然会报编译错误。