首頁 > 軟體

Swift繼承Inheritance淺析介紹

2022-08-24 14:01:21

繼承(Inheritance)

1、值型別(列舉、結構體)不支援繼承,只有類支援繼承

2、沒有父類別的類,稱為:基礎類別

Swift並沒有像OC、Java那樣的規定:任何類最終都要繼承自某個基礎類別。

3、子類可以重寫父類別的下標、方法、屬性,重寫必須加上override關鍵字。

記憶體結構

class Animal {
    var age = 0
}
class Dog : Animal {
    var weight = 0
}
class ErHa : Dog {
    var iq = 0
}
let a = Animal()
a.age = 10

看一下a需要多少記憶體,a是堆空間的,所以必然是16的倍數,最前面有8個位元組拿來放型別資訊,第二個8個位元組放參照計數相關的東西,再往後8個位元組才是放age,總共用到的是24個位元組,但是需要保證是16的倍數,所以是32個位元組。

Dog因為有繼承,所以等價於

class Animal {
    var age = 0
}
class Dog : Animal {
    var weight = 0
}
class Dog {
    var age = 0
    var weight = 0
}
class ErHa : Dog {
    var iq = 0
}
let d = Dog()
d.age = 10
d.weight = 20

d物件裡面有兩個屬性,age和weight,各佔8個位元組,並且一般來說父類別的屬性記憶體靠前,d物件也佔用32個位元組,第一塊是型別相關的8個位元組,第二塊是參照計數相關的8個位元組,第三塊是儲存age的8個位元組,第四塊是儲存weight的8個位元組。

同樣的一個ErHa物件要有24個位元組儲存age、weight、iq,另外還有前面的16個位元組,所以是40個位元組,但是要保證是16的倍數,所以就是48。

重寫實體方法、下標

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
    }
}
var anim: Animal
anim = Animal()
//Animal speak
anim.speak()
//6
print(anim[6])
anim = Cat()
//Animal speak
//Cat speak
anim.speak()
// 7
print(anim[6])

重寫型別方法、下標

1、被class修飾的型別方法、下標,允許被子類重寫

2、被static修飾的型別方法、下標,不允許被子類重寫

class Animal {
    class func speak() {
        print("Animal speak")
    }
    class subscript(index: Int) -> Int {
        return index
    }
}
class Cat : Animal {
    override class func speak() {
        super.speak()
        print("Cat speak")
    }
    override class subscript(index: Int) -> Int {
        return super[index] + 1
    }
}

static修飾的型別方法、下標重寫報錯

子類重寫可以用static修飾,只不過不能再繼續被重寫了

重寫屬性

1、子類可以將父類別的屬性(儲存、計算)重寫為計算屬性

2、子類不可以將父類別屬性重寫為儲存屬性

3、只能重寫var屬性,不能重寫let屬性

4、重寫時,屬性名、型別要一致

5、子類重寫後的屬性許可權,不能小於父類別屬性的許可權

如果父類別屬性是唯讀的,那麼子類重寫後的屬性可以是唯讀的,也可以是可讀寫的。

如果父類別屬性是可讀寫的,那麼子類重寫後的屬性也必須是可讀寫的。

重寫範例屬性

class Circle {
    var radius: Int = 0
    var diameter: Int {
        set {
            print("Circle setDiameter")
            radius = newValue / 2
        }
        get {
            print("Circle getDiameter")
            return radius * 2
        }
    }
}
class SubCircle: Circle {
    override var radius: Int {
        set {
            print("SubCircle setRadius")
            super.radius = newValue > 0 ? newValue : 0
        }
        get {
            print("SubCircle getRadius")
            return super.radius
        }
    }
    override var diameter: Int {
        set {
            print("SubCircle setDiameter")
            super.diameter = newValue > 0 ? newValue : 0
        }
        get {
            print("SubCircle getDiameter")
            return super.diameter
        }
    }
}
var circle = SubCircle()
circle.radius = 6
//SubCircle setRadius
print(circle.diameter)
//SubCircle getDiameter
//Circle getDiameter
//SubCircle getRadius
//12
circle.diameter = 20
//SubCircle setDiameter
//Circle setDiameter
//SubCircle setRadius
print(circle.radius)
//SubCircle getRadius
//10

重寫型別屬性

1、被class修飾的計算型別屬性,可以被子類重寫

儲存型別屬性只能用static來修飾。

2、被static修飾的型別屬性(儲存、計算),不可以被子類重寫

屬性觀察器

1、可以在子類中為父類別屬性(除了唯讀計算屬性、let屬性)增加屬性觀察器

class Circle {
    var radius: Int = 1
}
class SubCircle: Circle {
    override var radius: Int {
        willSet {
            print("SubCircle willSetRadius", newValue)
        }
        didSet {
            print("SubCircle didSetRadius", oldValue, radius)
        }
    }
}
var circle = SubCircle()
circle.radius = 10
//SubCircle willSetRadius 10
//SubCircle didSetRadius 1 10

2、父類別和子類中都有屬性觀察器

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()
circle.radius = 10
//SubCircle willSetRadius 10
//Circle willSetRadius 10
//Circle didSetRadius 1 10
//SubCircle didSetRadius 1 10

3、子類是可以給父類別中的計算屬性增加屬性觀察器的。

class Circle {
    class var radius: Int {
        set {
            print("Circle setRadius", newValue)
        }
        get {
            print("Circle getRadius")
            return 20
        }
    }
}
class SubCircle: Circle {
    override static var radius: Int {
        willSet {
            print("SubCircle willSetRadius", newValue)
        }
        didSet {
            print("SubCircle didSetRadius", oldValue, radius)
        }
    }
}
SubCircle.radius = 10
//        Circle getRadius (oldValue)
//        SubCircle willSetRadius 10
//        Circle setRadius 10
//        Circle getRadius (radius)
//        SubCircle didSetRadius 20 20

final

1、被final修飾的方法、下標、屬性,禁止被重寫

2、被final修飾的類,禁止被繼承

到此這篇關於Swift繼承Inheritance淺析介紹的文章就介紹到這了,更多相關Swift繼承內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!


IT145.com E-mail:sddin#qq.com