本文首发于我的博客html
同其余语言同样,Swift中也是有继承的git
override
关键字Animal类
,其中Dog 类
继承Animal类
,其中ErHa 类
继承Dog类
class Animal {
var age = 0
}
class Dog : Animal {
var weight = 0
}
class ErHa : Dog {
var iq = 0
}
复制代码
Animal类
的内存以下代码查看类的内存大小和相应内存的值,须要用到工具Mems 关于的使用能够看Swift之枚举一文,或者直接看github上Mems的说明github
let a = Animal()
a.age = 5
print(Mems.size(ofRef: a))
print(Mems.memStr(ofRef: a))
复制代码
Animal类
的内存大小和内容,分别为编程
32
0x00000001000084d8 //存放类的相关信息
0x0000000000000002 //引用技术
0x0000000000000005 // age数值5
0x0000000000000000 //没用到
复制代码
Animal类
的内存分析Dog类
的内存以下代码查看类的内存大小和相应内存的值bash
let d = Dog()
d.age = 6
d.weight = 7
print(Mems.size(ofRef: d))
print(Mems.memStr(ofRef: d))
复制代码
Dog类
的内存大小和内容,分别为app
32
0x0000000100008588 //存放类的相关信息
0x0000000000000002 //引用技术
0x0000000000000006 // age数值6
0x0000000000000007 //weight的值7
复制代码
Dog类
的内存分析ErHa类
的内存以下代码查看类的内存大小和相应内存的值ide
let e = ErHa()
e.age = 8
e.weight = 9
e.iq = 10
print(Mems.size(ofRef: e))
print(Mems.memStr(ofRef: e))
复制代码
ErHa类
的内存大小和内容,分别为工具
48
0x0000000100008658 //存放类的相关信息
0x0000000000000002 //引用技术
0x0000000000000008 // age数值8
0x0000000000000009 //weight的值9
0x000000000000000a //iq的值10
0x0000000000000000 //内存对齐增长的,没用到
复制代码
ErHa类
的内存分析子类能够重写父类的实例方法、下标post
有个Animal
类ui
class Animal {
func speak() {
print("Animal speak")
}
subscript(index: Int) -> Int {
return index
}
}
复制代码
子类Cat
继承Animal
,若是重写示例方法,下标,必须用关键字override
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
}
}
复制代码
上面的实例方法、下标。若是是类型方法、下标的话,有些许不一样,
eg
class Animal {
//static修饰
static func speak() {
print("Animal speak")
}
// class修饰
class subscript(index: Int) -> Int {
return index
}
}
print(Animal[6])
class Cat : Animal {
//编译报错 Cannot override static method
override class func speak() {
super.speak()
print("Cat speak")
}
//编译成功
override class subscript(index: Int) -> Int {
return super[index] + 1
}
}
复制代码
如上面的代码所示static
修饰的时候,子类重写,直接报错Cannot override static method
。而class
修饰时候,编译正常
重写类型属性,比较简单,不作赘述
注意的是:若是子类把父类的存储属性int 重写为计算属性,子类中依然有8个字节存储该int属性
再次须要注意的是,和重写类型方法、下标相似 若是重写类型属性
eg,子类SubCircle
给父类Circle
的存储属性radius
增长属性观察器
class Circle {
var radius: Int = 1
}
class SubCircle : Circle {
override var radius: Int {
willSet {
print("SubCircle willSetRadius", newValue)
}
didSet {
print("SubCircle didSetRadius", oldValue, radius)
}
}
}
复制代码
注意点:子类增长属性观察器以后,依然是存储属性
eg:
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()
复制代码
输出为
SubCircle willSetRadius 10
Circle willSetRadius 10
Circle didSetRadius 1 10
SubCircle didSetRadius 1 10
复制代码
以下代码
class Circle {
var radius: Int {
set {
print("Circle setRadius", newValue)
}
get {
print("Circle getRadius")
return 20
}
}
}
class SubCircle : Circle {
override var radius: Int {
willSet {
print("SubCircle willSetRadius", newValue)
}
didSet {
print("SubCircle didSetRadius", oldValue, radius)
}
}
}
复制代码
使用
var circle = SubCircle()
circle.radius = 10
复制代码
输出结果为
Circle getRadius
SubCircle willSetRadius 10
Circle setRadius 10
Circle getRadius
SubCircle didSetRadius 20 20
复制代码
输出结果分析
circle.radius = 10
的时候,先获取了oldValue
,调用父类的get
输出Circle getRadius
willSetRadius
准备赋值setRadius
print("SubCircle didSetRadius", oldValue, radius)
以前,要先获取radius
的值,因此,须要先执行父类的getRadius
didSet
方法参考资料: