在 Swift 中,类可以调用和访问超类的方法,属性和下标脚本(subscripts),并且可以重写(override)这些方法,属性和下标脚本来优化或修改它们的行为。Swift 会检查你的重写定义在超类中是否有匹配的定义,以此确保你的重写行为是正确的。
可以为类中继承来的属性添加属性观察器(property observers),这样一来,当属性值改变时,类就会被通知到。可以为任何属性添加属性观察器,无论它原本被定义为存储型属性(stored property)还是计算型属性(computed property)。
//定义一个基类(父类)
class Founder{
var name = ""
var money:Int?
var company: String = ""
var title: String{
get{
return "\(company) CEO"
}
}
func say(){
print("我叫\(name),是\(company)的创始人,我拥有\(money!)亿的财富!")
}
}
//定义一个子类
class Heir:Founder{
//重写属性观察器.
override var name:String{
didSet{
print("\(name)")
}
}
//重写只读属性
override var title: String{
return "\(company) 二代 CEO"
}
//重写方法
override func say(){
super.say()
print("我叫\(name),是\(company)的下一代负责人,我拥有\(money!)亿的财富!")
}
}
func testOne(){
let father = Founder()
father.name = "jack"
father.company = "上帝集团"
father.money = 1000
father.say()
let son = Heir()
son.name = "jey"
son.company = "上帝集团"
son.money = 1000
son.say()
}
防止重写
可以通过把方法,属性或下标脚本标记为final
来防止它们被重写,只需要在声明关键字前加上final
特性即可。(例如:final var
,final func
,final class func
, 以及final subscript
)
如果重写了final
方法,属性或下标脚本,在编译时会报错。在类扩展中的方法,属性或下标脚本也可以在扩展的定义里标记为 final。
可以通过在关键字class
前添加final
特性(final class
)来将整个类标记为 final 的,这样的类是不可被继承的,任何子类试图继承此类时,在编译时会报错。
class Founder{
var name = ""
var money:Int?
var company: String = ""
final var title: String{ //这里使用了final,子类将不能够重写.重写会报错
get{
return "\(company) CEO"
}
}
func say(){
print("我叫\(name),是\(company)的创始人,我拥有\(money!)亿的财富!")
}
}
权限控制
给一段代码加上final就意味着编译器向你作出保证,这段代码不会再被修改;同时,这也意味着你认为这段代码已经完备并且没有再被进行继承或重写的必要,因此这往往会是一个需要深思熟虑的决定。
一般来说,不希望被继承和重写会有这几种情况:
1)类或者方法的功能确实已经完备了。
2)子类继承和修改是一件危险的事情。即在子类继承或重写某些方法后可能做一些破坏性的事情,导致子类或者父类部分也无法正常工作的情况。
3)为了父类中某些代码一定会被执行。有时候父类中有一些关键代码是在被继承重写后必须执行的 (比如状态配置,认证等等),否则将 导致运行时候的错误
性能考虑
使用final的另一个重要理由是可能带来的性能改善。因为编译器能够从final中获取额外的信息,因此可以对类或者方法调用进行额外的优化处理。但是这个优势在实际表现中可能带来的好 处其实就算与 Objective-C 的动态派发相比也十分有限,因此在项目还有其他方面可以优化 (一般 说会是算法或者图形相关的内容导致性能瓶颈) 的情况下,并不建议使用将类或者方法转换的方式来追求性能的提升。