在线时间:8:00-16:00
迪恩网络APP
随时随地掌握行业动态
扫描二维码
关注迪恩网络微信公众号
继承语法
继承是面向对象最显著的一个特性, 继承是从已经有的类中派生出新的类 新的类能够继承已有类的属性和方法, 并能扩展新的能力 术语: 基类(父类, 超类), 派生类(子类, 继承类) 语法: class 子类: 父类{ }
继承优点: 代码重用
继承缺点: 增加程序耦合度, 父类改变会影响子类
注意:Swift和OC一样没有多继承
class Man {
var name:String = "qbs"
var age: Int = 10
func sleep(){ print("睡觉") } }
class SuperMan: Man {
var power:Int = 100 funs fly(){
子类可以继承父类的属性
print("飞 \(name) \(age)")
} } var m = Man() m.sleep()
父类不可以使用子类的方法
m.fly()
输出结果: 睡觉
var sm = SuperMan()
子类可以继承父类的方法
sm.sleep()
sm.fly()
输出结果:
睡觉
飞 qbs 10
super关键字:
派生类中可以通过super关键字来引用父类的属性和方法
class Man {
var name:String = "qbs"
var age: Int = 30
func sleep(){ print("睡觉") } } class SuperMan: Man { var power:Int = 100 func eat() { print("吃饭") } func fly(){
子类可以继承父类的属性
print("飞 \(super.name) \(super.age)")
} func eatAndSleep() { eat() super.sleep()
如果没有写super, 那么会先在当前类中查找, 如果找不到再去父类中查找
如果写了super, 会直接去父类中查找
}
} var sm = SuperMan() sm.eatAndSleep()
输出结果:
吃饭
睡觉
方法重写: override
重写父类方法, 必须加上override关键字
class Man {
var name:String = "qbs"
var age: Int = 10
func sleep(){ print("父类睡觉") } } class SuperMan: Man { var power:Int = 100
override关键字主要是为了明确表示重写父类方法,
所以如果要重写父类方法, 必须加上override关键字
override func sleep() {
sleep() 不能这样写, 会导致递归
super.sleep()
print("子类睡觉") } func eat() { print("吃饭") } func fly(){
子类可以继承父类的属性
print("飞 \(super.name) \(super.age)")
} func eatAndSleep() { eat() sleep() } } var sm = SuperMan()
通过子类调用, 优先调用子类重写的方法
sm.sleep()
输出结果:
父类睡觉
子类睡觉
sm.eatAndSleep()
输出结果:
吃饭
父类睡觉 子类睡觉
重写属性
无论是存储属性还是计算属性, 都只能重写为计算属性
class Man {
var name:String = "qbs" 存储属性
var age: Int { 计算属性
get{
return 30 } set{ print("man new age \(newValue)") } } func sleep(){ print("睡觉") } } class SuperMan: Man { var power:Int = 100
可以将父类的存储属性重写为计算属性
但不可以将父类的存储属性又重写为存储属性, 因为这样没有意义
override var name:String = "zs"
override var name:String{
get{ return "zs" } set{ print("SuperMan new name \(newValue)") } }
可以将父类的计算属性重写为计算属性, 同样不能重写为存储属性
override var age: Int { 计算属性
get{
return 30 } set{ print("superMan new age \(newValue)") } } } let sm = SuperMan()
通过子类对象来调用重写的属性或者方法, 肯定会调用子类中重写的版本
sm.name = "xxxoo"
sm.age = 50
输出结果:
SuperMan new name xxxoo
superMan new age 50
重写属性的限制
1.读写计算属性/存储属性, 是否可以重写为只读计算属性? (权限变小)不可以
2.只读计算属性, 是否可以在重写时变成读写计算属性? (权限变大)可以
class Man {
var name:String = "qbs" 存储属性
var age: Int { 计算属性
get{
return 30 } set{ print("man new age \(newValue)") } } func sleep(){ print("睡觉") } } class SuperMan: Man { var power:Int = 100 override var name:String{ get{ return "zs" } set{ print("SuperMan new name \(newValue)") } }
override var age: Int { 计算属性
get{
return 30 } set{ print("superMan new age \(newValue)") } } }
重写属性观察器
只能给非lazy属性的变量存储属性设定属性观察器,
不能给计算属性设置属性观察器,给计算属性设置属性观察器没有意义 属性观察器限制: 1.不能在子类中重写父类只读的存储属性 2.不能给lazy的属性设置属性观察器
class Man {
var name: String = "qbs"
var age: Int = 0 { 存储属性
willSet{
print("super new \(newValue)") } didSet{ print("super new \(oldValue)") } } var height:Double{ get{ print("super get") return 10.0 } set{ print("super set") } } } class SuperMan: Man {
可以在子类中重写父类的存储属性为属性观察器
override var name: String {
willSet{ print("new \(newValue)") } didSet{ print("old \(oldValue)") } }
可以在子类中重写父类的属性观察器
override var age: Int{
willSet{ print("child new \(newValue)") } didSet{ print("child old \(oldValue)") } }
可以在子类重写父类的计算属性为属性观察器
override var height:Double{
willSet{ print("child height") } didSet{ print("child height") } } }
var m = SuperMan()
m.age = 55
输出结果:
child new 55
super new 55 super new 0 child old 0
print(m.age)
输出结果: 55
m.height = 20.0
输出结果:
super get
child height super set child height
final关键字
利用final关键字防止重写
final关键字既可以修饰属性, 也可以修饰方法, 并且还可以修饰类 被final关键字修饰的属性和方法不能被重写 被final关键字修饰的类不能被继承
final class Man {
final var name: String = "qbs"
final var age: Int = 0 { // 存储属性
willSet{ print("super new \(newValue)") } didSet{ print("super new \(oldValue)") } } final var height:Double{ get{ print("super get") return 10.0 } set{ print("super set") } } final func eat(){ print("吃饭") } }
|
请发表评论