Swift語法 Swift5 【11 - 繼承】


  • 作者: Liwx
  • 郵箱: 1032282633@qq.com
  • 源碼: 需要源碼的同學(xué), 可以在評論區(qū)留下您的郵箱

iOS Swift 語法 底層原理內(nèi)存管理分析 專題:【iOS Swift5語法】

00 - 匯編
01 - 基礎(chǔ)語法
02 - 流程控制
03 - 函數(shù)
04 - 枚舉
05 - 可選項
06 - 結(jié)構(gòu)體和類
07 - 閉包
08 - 屬性
09 - 方法
10 - 下標(biāo)
11 - 繼承
12 - 初始化器init
13 - 可選項


目錄

  • 01-繼承(Inheritance)
  • 02-內(nèi)存結(jié)構(gòu)
  • 03-重寫實例方法、下標(biāo)
  • 04-重寫類型方法、下標(biāo)
  • 05-重寫屬性
  • 06-重寫實例屬性
  • 07-重寫類型屬性
  • 08-屬性觀察器(子類為父類存儲屬性添加屬性觀察器)
  • 09-屬性觀察器(子類和父類都實現(xiàn)屬性觀察器)
  • 10-屬性觀察器(子類為父類計算屬性添加屬性觀察器)
  • 11-屬性觀察器(子類為父類類型計算屬性添加屬性觀察器)
  • 12-final

01-繼承(Inheritance)

  • 值類型(枚舉、結(jié)構(gòu)體)不支持繼承, 只有類支持繼承
  • 沒有父類的類,稱為: 基類
    • Swift并沒有像OC、Java那樣的規(guī)定: 任何類最終都要繼承自某個基類
  • 子類可以重寫父類的下標(biāo)、方法、屬性, 重寫必須加上override關(guān)鍵字

02-內(nèi)存結(jié)構(gòu)

  • 前16個字節(jié)是用來存放類型信息引用計數(shù)

class Animal {
    var age = 0
}

class Dog : Animal {
    var weight = 0
}

class ErHa : Dog {
    var iq = 0
}

let a = Animal()
a.age = 10
print(Mems.size(ofRef: a))  // 32
/*
 0x00000001000073c8  // 類型信息
 0x0000000000000002  // 引用計數(shù)
 0x000000000000000a  // 存儲屬性 age
 0x0000000000000000  // 未使用
 */
print(Mems.memStr(ofRef: a))

let d = Dog()
d.age = 10
d.weight = 20
let dd = d
print(dd)
print(Mems.size(ofRef: d))  // 32
/*
 0x0000000100007478  // 類型信息
 0x0000000000000002  // 引用計數(shù)
 0x000000000000000a  // 存儲屬性 age
 0x0000000000000014  // 存儲屬性weight
 */
print(Mems.memStr(ofRef: d))

let e = ErHa()
e.age = 10
e.weight = 20
e.iq = 30
print(Mems.size(ofRef: e))  // 48
/*
 0x0000000100008548  // 類型信息
 0x0000000000000002  // 引用計數(shù)
 0x000000000000000a  // 存儲屬性 age
 0x0000000000000014  // 存儲屬性weight
 0x000000000000001e  // 存儲屬性iq
 0x0000000000343130  // 未使用
 */

03-重寫實例方法、下標(biāo)

  • 重寫實例方法下標(biāo) override修飾下標(biāo)方法subscript
    • 調(diào)用父類的下標(biāo)方法subscript: super[index]
class Animal {
    func speak() {
        print("Animal speak")
    }
    subscript(index: Int) ->Int {
        return index
    }
}
var anim: Animal
anim = Animal()
anim.speak()    // Animal speak
print(anim[6])  // 6

class Cat : Animal {
    override func speak() {
        print("Cat speak")
    }
    override subscript(index: Int) -> Int {
        return super[index] + 1     // 調(diào)用父類下標(biāo)方法subscript
    }
}

anim = Cat()    // 多態(tài), 父類指針指向子類對象
anim.speak()    // Cat speak
print(anim[6])  // 7

04-重寫類型方法、下標(biāo)

  • class修飾的類型方法、下標(biāo), 允許被子類重寫
  • static修飾的類型方法、下標(biāo), 不允許被子類重寫
class Animal {
    class func speak() {
        print("Animal speak")
    }
    class subscript(index: Int) -> Int {
        return index
    }
}

Animal.speak()      // Animal speak
print(Animal[6])    // 6

class Cat : Animal {
    override class func speak() {
        print("Cat speak")
    }
    override class subscript(index: Int) -> Int {  // 如果父類用static修飾下標(biāo)方法subscript, 報錯: Cannot override static subscript
        return super[index] + 1
    }
}

Cat.speak()         // Cat speak
print(Cat[6])       // 7

05-重寫屬性

  • 子類可以將父類的屬性(存儲、計算)重寫為計算屬性
  • 子類不可以將父類的屬性重寫為存儲屬性
  • 只能重寫var屬性, 不能重寫let屬性
  • 重寫時,屬性名、類型要一致
    子類重寫后的屬性權(quán)限不能小于 父類屬性的權(quán)限
    • 如果父類屬性是只讀的,那么子類重寫后的屬性可以是只讀的、也可以是可讀寫
    • 如果父類屬性是可讀寫的,那么子類重寫后的屬性也必須是可讀寫

06-重寫實例屬性

  • 重寫實例屬性, 存儲,計算屬性重寫
class Circle {
    var radius: Int = 0
    var diameter: Int {
        set {
            print("Circle setDiameter")
            radius = newValue / 2
        }
        get {
            print("Circle getDiameter")
            return radius * 2
        }
    }
}

var circle: Circle
circle = Circle()
circle.radius = 6

// Circle getDiameter
// 12
print(circle.diameter)

// Circle setDiameter
circle.diameter = 20
// 10
print(circle.radius)


class SubCircle : Circle {
    override var radius: Int {
        set {
            print("SubCircle setRadius")
            super.radius = newValue
        }
        get {
            print("SubCircle getRadius")
            return super.radius
        }
    }
    
    override var diameter: Int {
        set {
            print("SubCircle setDiameter")
            super.diameter = newValue
        }
        get {
            print("SubCircle getDiameter")
            return super.diameter
        }
    }
}

circle = SubCircle()
// SubCircle setRadius
circle.radius = 6

// SubCircle getDiameter
// Circle getDiameter
// SubCircle getRadius
// 12
print(circle.diameter)

// SubCircle setDiameter
// Circle setDiameter
// SubCircle setRadius
circle.diameter = 20

// SubCircle getRadius
// 10
print(circle.radius)

07-重寫類型屬性

  • class修飾的計算類型屬性, 可以被子類重寫
  • static修飾的類型屬性(存儲, 計算), 不可以被子類重寫
class Circle {
    static var radius: Int = 0
    class var diameter: Int {
        set {
            print("Circle setDiameter")
            radius = newValue / 2
        }
        get {
            print("Circle getDiameter")
            return radius * 2
        }
    }
}

class SubCircle : Circle {
//    override var radius: Int {  static修飾的類型屬性不能重寫 // error: property does not override any property from its superclass
//        set {
//            super.radius = 2
//        }
//        get {
//            return 10
//        }
//    }
    override class var diameter: Int {
        set {
            print("SubCircle setDiameter")
            super.diameter = newValue > 0 ? newValue : 0
        }
        get {
            print("SubCircle getDiameter")
            return super.diameter
        }
    }
}

Circle.radius = 6
// 6
print(Circle.radius)
// Circle getDiameter
Circle.diameter = 20
// 10
print(Circle.radius)

print("----")

SubCircle.radius = 6
// SubCircle getDiameter
// Circle getDiameter
// 12
print(SubCircle.diameter)
// SubCircle setDiameter
// Circle setDiameter
SubCircle.diameter = 20
// 10
print(SubCircle.radius)

08-屬性觀察器(子類為父類存儲屬性添加屬性觀察器)

  • 可以在子類中為父類屬性(除了只讀計算屬性let屬性)增加屬性觀察器
class Circle {
    var radius: Int = 1
    
    let count: Int = 1
    var diameters: Int {    // 只讀計算屬性
        radius * 2
    }
}
class SubCircle : Circle {
    override var radius: Int {
        willSet {
            print("SubCircle willSetRadius", newValue)
        }
        didSet {
            print("SubCircle didSetRadius", oldValue, radius)
        }
    }
    
//    // 不能給let屬性重寫屬性觀察器
//    override let count: Int {   // error: 'let' declarations cannot be observing properties
//        willSet {
//            
//        }
//        didSet {
//        
//        }
//    }
  
//    // 不能給只讀計算屬性重寫屬性觀察器
//    override var diameters: Int { // error: cannot observe read-only property 'diameters'; it can't change
//        willSet {
//
//        }
//        didSet {
//
//        }
//    }
    
}
var circle = SubCircle()

// SubCircle willSetRadius 10
// SubCircle didSetRadius 1 10
circle.radius = 10

09-屬性觀察器(子類和父類都實現(xiàn)屬性觀察器)

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()
// SubCircle willSetRadius 10
// Circle willSetRadius 10
// Circle didSetRadius 1 10
// SubCircle didSetRadius 1 10
circle.radius = 10
  • 通過匯編觀察(子類和父類都實現(xiàn)屬性觀察器)
image.png
// 根據(jù)匯編代碼分析
set {
    call willSet
    // 真正設(shè)置值
    call super set
    call didSet
}

10-屬性觀察器(子類為父類計算屬性添加屬性觀察器)

  • 計算屬性沒占用內(nèi)存,所以使用Copy In Copy Out方式,先調(diào)用get方法復(fù)制一個副本
class Circle {
    var radius: Int {
        set {
            print("Circle setRadius", newValue)
        }
        get {
            print("Circle getRadius")
            return 20
        }
    }
}

class SubCircle : Circle {
    override var radius: Int {
        willSet {
            print("SubCircle willSet", newValue)
        }
        didSet {
            print("SubCircle didSet", oldValue, radius)
        }
    }
}

var circle = SubCircle()
// Circle getRadius      // 重寫計算屬性的屬性觀察器,計算屬性原本不占用內(nèi)存,所以會先調(diào)用get方法復(fù)制一個副本
// SubCircle willSet 10
// Circle setRadius 10
// Circle getRadius      // 此次打印get是因為父類的radius set方法中調(diào)用了newValue
// SubCircle didSet 20 20
circle.radius = 10

11-屬性觀察器(子類為父類類型計算屬性添加屬性觀察器)

  • Xcode 11.4.1 以下代碼報錯, 不知什么原因! 待排查
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 willSet", newValue)
        }
        didSet {
            print("SubCircle didSet", oldValue, radius)
        }
    }
}

// Circle getRadius   // oldValue是設(shè)置之前的值,所以在即將設(shè)置之前會調(diào)用getRadius來獲取設(shè)置之前的值
// SubCircle willSet 10
// Circle setRadius 10
// Circle getRadius
// SubCircle didSetRadius 20 20
SubCircle.radius = 10

12-final

  • final修飾的方法、下標(biāo)、屬性,禁止被重寫
class Circle {
    final func test() {
        print("Circle test")
    }
}

class SubCircle : Circle {
   // 不能重寫final修飾的方法
    override final func test() {    // error: instance method overrides a 'final' instance method
    }
}
  • final修飾的,禁止被繼承
final class Circle {
    var radius: Int = 0
}
// 不能繼承final修飾的類
class SubCircle : Circle {  // error: inheritance from a final class 'Circle'
}

iOS Swift 語法 底層原理內(nèi)存管理分析 專題:【iOS Swift5語法】

下一篇: 12 - 初始化器init
上一篇: 10 - 下標(biāo)


最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。
禁止轉(zhuǎn)載,如需轉(zhuǎn)載請通過簡信或評論聯(lián)系作者。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,316評論 6 531
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,481評論 3 415
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,241評論 0 374
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,939評論 1 309
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 71,697評論 6 409
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,182評論 1 324
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,247評論 3 441
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,406評論 0 288
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,933評論 1 334
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 40,772評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 42,973評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,516評論 5 359
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,209評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,638評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,866評論 1 285
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,644評論 3 391
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 47,953評論 2 373