重寫實例屬性
我們可以在子類中重寫從父類繼承來的屬性,屬性有實例屬性和靜態屬性之分,他們在具體實現也是不同的。
實例屬性的重寫一方面可以重寫getter和setter訪問器,另一方面可以重寫屬性觀察者。
計算靜態屬性需要使用getter和setter訪問器,而存儲屬性不需要。子類在繼承父類後,也可以通過getter和setter訪問器重寫父類的存儲屬性和計算屬性。
下面看一個示例:
class Person { var name: String //存儲屬性 var age: Int //存儲屬性 func description() -> String{ return "\(name) 年齡是: \(age)" } init (name: String, age: Int){ self.name = name self.age = age } } class Student: Person{ var school: String override var age: Int {//重寫屬性前面要添加override關鍵字 get { return super.age } set { super.age = newValue < 8 ? 8: newValue } } convenience init() { self.init(name: "Tony", age: 18, school: "清華大學") } init (name: String, age: Int,school: String) { self.school = school super.init(name: name, age: age) } } let student1 = Student() print("學生年齡:\(student1.age)") student1.age = 6 print("學生年齡:\(student1.age)") 從屬性重寫可見,子類本身並不存儲數據,數據是存儲在父類的存儲屬性中的。 以上示例是重寫屬性getter和setter訪問器,我們還可以重寫屬性觀察者,代碼如下:
class Person { var name: String var age: Int func description() -> String{ return "\(name) 年齡是: \(age)" } init (name: String, age: Int){ self.name = name self.age = age } } class Student: Person{ var school: String override var age: Int { //重寫了age屬性觀察者 willSet { //如果只關注修改之前的調用,可以只重寫willSet觀察者 print("學生年齡新值:\(newValue)") } didSet{ //如果只關注修改之後的調用,可以只重寫didSet觀察者 print("學生年齡舊值:\(oldValue)") } } convenience init() { self.init(name: "Tony", age: 18, school: "清華大學") } init (name: String, age: Int,school: String) { self.school = school super.init(name: name, age: age) } } let student1 = Student() print("學生年齡:\(student1.age)") Student1.age = 6 print("學生年齡:\(student1.age)")
代碼Student1.age = 6修改了age屬性,修改前後的輸出結果如下:
學生年齡新值:6
學生年齡舊值:18
重寫靜態屬性
在類中靜態屬性定義使用class或static關鍵字,但是使用哪一個要看子類中是否重寫該屬性。class修飾的屬性可以被重寫,static關鍵字就不能被重寫。
示例代碼如下:
class Account { var amount: Double = 0.0 // 賬戶金額 var owner: String = "" //賬戶名 var interestRate: Double = 0.0668 //利率 //class不能換成static class var staticProp: Double { //靜態屬性staticProp return 0.0668 * 1_000_000 } var instanceProp: Double { return self.interestRate *self.amount } } class TermAccount: Account { //class換成static override class var staticProp: Double { //重寫靜態屬性staticProp return 0.0700 * 1_000_000 } } //訪問靜態屬性 print(Account.staticProp) print(TermAccount.staticProp)
由於要被重寫所以代碼class var staticProp: Double 中的class不能換成static。代碼overrideclass var staticProp: Double中的靜態屬性staticProp可以使用class或static,除非在TermAccount的子類中重寫屬性staticProp。