Swift中的继承(Inheritance)
2022-07-04 本文已影响0人
扑腾的蛾子
继承(Inheritance)
1、值类型(枚举、结构体)不支持继承,只有类支持继承
2、没有父类的类,称为:基类
Swift并没有像OC、Java那样的规定:任何类最终都要继承自某个基类。
3、子类可以重写父类的下标、方法、属性,重写必须加上override关键字。
内存结构
class Animal {
var age = 0
}
class Dog : Animal {
var weight = 0
}
class ErHa : Dog {
var iq = 0
}
let a = Animal()
a.age = 10
看一下a需要多少内存,a是堆空间的,所以必然是16的倍数,最前面有8个字节拿来放类型信息,第二个8个字节放引用计数相关的东西,再往后8个字节才是放age,总共用到的是24个字节,但是需要保证是16的倍数,所以是32个字节。
Dog因为有继承,所以等价于
class Animal {
var age = 0
}
class Dog : Animal {
var weight = 0
}
class Dog {
var age = 0
var weight = 0
}
class ErHa : Dog {
var iq = 0
}
let d = Dog()
d.age = 10
d.weight = 20
d对象里面有两个属性,age和weight,各占8个字节,并且一般来说父类的属性内存靠前,d对象也占用32个字节,第一块是类型相关的8个字节,第二块是引用计数相关的8个字节,第三块是存储age的8个字节,第四块是存储weight的8个字节。
同样的一个ErHa对象要有24个字节存储age、weight、iq,另外还有前面的16个字节,所以是40个字节,但是要保证是16的倍数,所以就是48。
重写实例方法、下标
class Animal {
func speak() {
print("Animal speak")
}
subscript(index: Int) -> Int {
return index
}
}
class Cat : Animal {
override func speak() {
super.speak()
print("Cat speak")
}
override subscript(index: Int) -> Int {
return super[index] + 1
}
}
var anim: Animal
anim = Animal()
//Animal speak
anim.speak()
//6
print(anim[6])
anim = Cat()
//Animal speak
//Cat speak
anim.speak()
// 7
print(anim[6])
重写类型方法、下标
1、被class修饰的类型方法、下标,允许被子类重写
2、被static修饰的类型方法、下标,不允许被子类重写
class Animal {
class func speak() {
print("Animal speak")
}
class subscript(index: Int) -> Int {
return index
}
}
class Cat : Animal {
override class func speak() {
super.speak()
print("Cat speak")
}
override class subscript(index: Int) -> Int {
return super[index] + 1
}
}
static修饰的类型方法、下标重写报错
子类重写可以用static修饰,只不过不能再继续被重写了
重写属性
1、子类可以将父类的属性(存储、计算)重写为计算属性
2、子类不可以将父类属性重写为存储属性
3、只能重写var属性,不能重写let属性
4、重写时,属性名、类型要一致
5、子类重写后的属性权限,不能小于父类属性的权限
如果父类属性是只读的,那么子类重写后的属性可以是只读的,也可以是可读写的。
如果父类属性是可读写的,那么子类重写后的属性也必须是可读写的。
重写实例属性
class Circle {
var radius: Int = 0
var diameter: Int {
set {
print("Circle setDiameter")
radius = newValue / 2
}
get {
print("Circle getDiameter")
return radius * 2
}
}
}
class SubCircle: Circle {
override var radius: Int {
set {
print("SubCircle setRadius")
super.radius = newValue > 0 ? newValue : 0
}
get {
print("SubCircle getRadius")
return super.radius
}
}
override var diameter: Int {
set {
print("SubCircle setDiameter")
super.diameter = newValue > 0 ? newValue : 0
}
get {
print("SubCircle getDiameter")
return super.diameter
}
}
}
var circle = SubCircle()
circle.radius = 6
//SubCircle setRadius
print(circle.diameter)
//SubCircle getDiameter
//Circle getDiameter
//SubCircle getRadius
//12
circle.diameter = 20
//SubCircle setDiameter
//Circle setDiameter
//SubCircle setRadius
print(circle.radius)
//SubCircle getRadius
//10
重写类型属性
1、被class修饰的计算类型属性,可以被子类重写
存储类型属性只能用static来修饰。
2、被static修饰的类型属性(存储、计算),不可以被子类重写
属性观察器
1、可以在子类中为父类属性(除了只读计算属性、let属性)增加属性观察器
class Circle {
var radius: Int = 1
}
class SubCircle: Circle {
override var radius: Int {
willSet {
print("SubCircle willSetRadius", newValue)
}
didSet {
print("SubCircle didSetRadius", oldValue, radius)
}
}
}
var circle = SubCircle()
circle.radius = 10
//SubCircle willSetRadius 10
//SubCircle didSetRadius 1 10
2、父类和子类中都有属性观察器
class Circle {
var radius: Int = 1 {
willSet {
print("Circle willSetRadius", newValue)
}
didSet {
print("Circle didSetRadius", oldValue, radius)
}
}
}
class SubCircle: Circle {
override var radius: Int {
willSet {
print("SubCircle willSetRadius", newValue)
}
didSet {
print("SubCircle didSetRadius", oldValue, radius)
}
}
}
var circle = SubCircle()
circle.radius = 10
//SubCircle willSetRadius 10
//Circle willSetRadius 10
//Circle didSetRadius 1 10
//SubCircle didSetRadius 1 10
3、子类是可以给父类中的计算属性增加属性观察器的。
class Circle {
class var radius: Int {
set {
print("Circle setRadius", newValue)
}
get {
print("Circle getRadius")
return 20
}
}
}
class SubCircle: Circle {
override static var radius: Int {
willSet {
print("SubCircle willSetRadius", newValue)
}
didSet {
print("SubCircle didSetRadius", oldValue, radius)
}
}
}
SubCircle.radius = 10
// Circle getRadius (oldValue)
// SubCircle willSetRadius 10
// Circle setRadius 10
// Circle getRadius (radius)
// SubCircle didSetRadius 20 20
final
1、被final修饰的方法、下标、属性,禁止被重写
2、被final修饰的类,禁止被继承