Swift教程之繼承

繼承

類可以從另一個(gè)類繼承方法、屬性和其他特性,繼承類稱為子類,被繼承類為其超類。Swift的類可以調(diào)用和訪問(wèn)超類的方法、屬性和下標(biāo),并能重寫這些版本以改進(jìn)或修改其行為。

類還可以將屬性觀察器添加到繼承的屬性,屬性觀察器可以添加到任何屬性。

<br />

定義基類

基類是不繼承任何一個(gè)類的類。

注意

Swift類不會(huì)從通用基類繼承,未指定超類的自定義類自動(dòng)成為基類。

以下為一個(gè)自定義Vehicle基類:

class Vehicle {
    var currentSpeed = 0.0
    var description: String {
        return "traveling at \(currentSpeed) miles per hour"
    }
    func makeNoise() {
        // do nothing - an arbitrary vehicle doesn't necessarily make a noise
    }
}

使用初始化器語(yǔ)法創(chuàng)建一個(gè)類實(shí)例:

let someVehicle = Vehicle()

訪問(wèn)其description屬性:

print("Vehicle: \(someVehicle.description)")
// Vehicle: traveling at 0.0 miles per hour

<br />

子類化

子類化是將一個(gè)新類置于現(xiàn)有類之上的行為,子類繼承現(xiàn)有類的特性,也可以向子類添加新特性。

繼承語(yǔ)法如下:

class SomeSubclass: SomeSuperclass {
    // subclass definition goes here
}

子類化Vehicle類:

class Bicycle: Vehicle {
    var hasBasket = false
}

let bicycle = Bicycle()
bicycle.hasBasket = true

bicycle.currentSpeed = 15.0
print("Bicycle: \(bicycle.description)")
// Bicycle: traveling at 15.0 miles per hour

子類繼續(xù)子類化:

class Tandem: Bicycle {
    var currentNumberOfPassengers = 0
}

let tandem = Tandem()
tandem.hasBasket = true
tandem.currentNumberOfPassengers = 2
tandem.currentSpeed = 22.0
print("Tandem: \(tandem.description)")
// Tandem: traveling at 22.0 miles per hour

<br />

重寫

子類可以提供自定義的從超類繼承來(lái)的實(shí)例方法、類型方法、實(shí)例屬性、類型屬性或下標(biāo),稱為重寫。

使用override關(guān)鍵字重寫超類的特性,沒(méi)有關(guān)鍵字的任何重寫會(huì)導(dǎo)致編譯錯(cuò)誤。

訪問(wèn)超類方法、屬性和下標(biāo)

當(dāng)子類重寫方法、屬性或下標(biāo)時(shí),可以將現(xiàn)有超類的實(shí)現(xiàn)用作重寫的一部分,即在超類實(shí)現(xiàn)基礎(chǔ)上重寫。這時(shí)使用super前綴訪問(wèn)方法、屬性或下標(biāo)的超類版本:

  • 名為someMethod()的重寫方法可以通過(guò)在重寫方法實(shí)現(xiàn)中調(diào)用super.someMetho()來(lái)調(diào)用someMethod()的超類版本。

  • 被稱為someProperty的被覆蓋的屬性可以在超級(jí)的gettersetter實(shí)現(xiàn)中以super.someProperty方式訪問(wèn)someProperty的超類版本。

  • someIndex的覆蓋下標(biāo)可以從覆蓋的下標(biāo)實(shí)現(xiàn)中訪問(wèn)與super [someIndex]相同的下標(biāo)的超類版本。

方法重寫

可以重寫繼承的實(shí)例或類型方法來(lái)提供子類中定制或替代方法的方法。

class Train: Vehicle {
    override func makeNoise() {
        print("Choo Choo")
    }
}

let train = Train()
train.makeNoise()
// Prints "Choo Choo"

屬性重寫

可以重寫繼承的實(shí)例或類型屬性,為該屬性提供自定義gettersetter,或添加屬性觀察器以啟用重寫屬性來(lái)觀察底層屬性值何時(shí)更改。

getter和setter屬性重寫

可以提供自定義的getter(和setter)來(lái)覆蓋任何繼承的屬性,必須始終聲明要覆蓋的屬性名和類型,以便編譯器檢查重寫是否與具有同名同類型的超類屬性匹配。

可以為超類的只讀屬性的重寫同時(shí)提供gettersetter修改為讀寫屬性,但不能將超類的讀寫屬性重寫為只讀屬性。

注意

當(dāng)重寫屬性提供了setter時(shí),還必須提供getter。若不想覆蓋getter修改繼承屬性的值,可以從getter中返回super.someProperty以維持超類屬性的getter實(shí)現(xiàn)。

以下為子類重寫超類description屬性并在超類實(shí)現(xiàn)基礎(chǔ)上添加新功能:

class Car: Vehicle {
    var gear = 1
    override var description: String {
        return super.description + " in gear \(gear)"
    }
}

let car = Car()
car.currentSpeed = 25.0
car.gear = 3
print("Car: \(car.description)")
// Car: traveling at 25.0 miles per hour in gear 3

屬性觀察器重寫

重寫屬性會(huì)講超類的屬性觀察器添加到繼承的屬性,不論該屬性的實(shí)現(xiàn)方法如何都將能夠在繼承屬性的值更改時(shí)收到通知。

注意

不能將屬性觀察器添加到繼承的常量存儲(chǔ)型屬性或只讀計(jì)算型屬性,因?yàn)檫@些屬性值不能更改。不能為同一屬性同時(shí)提供重寫的setter和重寫的屬性觀察器,若要監(jiān)聽屬性值的更改,并且已經(jīng)提供重寫的setter,則可以簡(jiǎn)單地監(jiān)聽重寫的setter中任何值的更改。

class AutomaticCar: Car {
    override var currentSpeed: Double {
        didSet {
            gear = Int(currentSpeed / 10.0) + 1
        }
    }
}

let automatic = AutomaticCar()
automatic.currentSpeed = 35.0
print("AutomaticCar: \(automatic.description)")
// AutomaticCar: traveling at 35.0 miles per hour in gear 4

<br />

防止重寫

通過(guò)將方法、屬性或下標(biāo)標(biāo)記為final防止被子類重寫,如final var,final func,final class funfinal subscript。

任何試圖在子類中重寫標(biāo)記為final的方法、屬性或下標(biāo)都將報(bào)告編譯錯(cuò)誤,也可以在擴(kuò)展中使用final關(guān)鍵字。

可以在類定義中的class關(guān)鍵字之前添加final修飾符防止該類被子類化,任何試圖子類化標(biāo)記為final的類都將報(bào)告編譯錯(cuò)誤。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容