從零學(xué)習(xí)Swift 09: 匯編分析多態(tài),初始化器

總結(jié)

在 swift 中,類和結(jié)構(gòu)體都可以定義方法.那么類和結(jié)構(gòu)體定義的方法在調(diào)用的時候有什么不同嗎?我們看看下面的代碼.

首先看看調(diào)用結(jié)構(gòu)體方法的匯編:

調(diào)用結(jié)構(gòu)體方法

可以看到調(diào)用結(jié)構(gòu)體方法時,方法的地址是固定的.

我們把struct改成class,再來看看他的的匯編:

調(diào)用類的方法

可以看到,方法地址不是固定的,而是通過一定計算得來的.因為class支持繼承,所以它的方法地址一定不是寫死的,而是通過計算得來的.

所以在項目中如果我們只是單純的調(diào)用方法,那我們用結(jié)構(gòu)體就可以了,從匯編代碼可以看出,結(jié)構(gòu)體的匯編代碼更少,流程更簡單,效率更快.

一: swift 多態(tài)的實現(xiàn)原理

下面我們就來研究 Swift 中,多態(tài)的實現(xiàn)原理.

示例代碼:


class Person{
    func eat(){
        print("Person eat")
    }
    func run(){
        print("Person run")
    }
    
    func sleep(){
        print("Person sleep")
    }
    
}

class Student: Person {
    override func eat() {
         print("Student eat")
    }
    override func run() {
        print("Student run")
    }
    func study(){
        print("Student stuty")
    }
}

var person = Person()

person = Student()
person.eat()
person.run()
person.sleep()

//運行結(jié)果:
Student eat
Student run
Person sleep

上面代碼,父類類型Person指向子類Student的實例對象,最后調(diào)用的的結(jié)果.這就是多態(tài),swfit 內(nèi)部是怎樣實現(xiàn)多態(tài)的呢?

我們還是通過匯編窺探一下:

我們知道person是一個指針變量,它里面存儲的是堆空間對象的地址值.從匯編代碼可以看到,調(diào)用eat()時,先把person中存儲的堆空間對象的地址值取出來,然后movq指令取出堆空間的前8個字節(jié),也就是類型信息的地址.然后通過一個偏移量找到eat()方法的地址,并調(diào)用的.

畫圖表示:

上圖就是子類在調(diào)用方法的時候是如何調(diào)用的流程,需要注意的是類型信息存儲在全局區(qū)

二: 初始化器

之前講過,初始化器的唯一原則就是:保證所有成員都有值.在 swift 中,初始化器可以分為兩種:1: 指定初始化器 , 2:便捷初始化器

1:指定初始化器

指定初始化器就是之前我們使用的初始化器,如:

編譯器生成的無參初始化器

或者我們自定義的初始化器:

自定義初始化器

以上兩種都是指定初始化器,需要注意的是,一旦我們自定義了初始化器,編譯器就不會為我們再生成初始化器了.

指定初始化器就是類的主要初始化器,可以理解為主線.一個類至少有一個指定初始化器,并且 swift 官方建議不要過多的指定初始化器,一個類通常只有一個指定初始化器.

1: 便捷初始化器

便捷初始化器,顧名思義就是為了便捷,為了方便,比如:

便捷初始化器

可以看到,我們可以根據(jù)自己的需求定義多個便捷初始化器.
注意: 在便捷初始化器內(nèi)部必須調(diào)用指定初始化器,不然會報錯

便捷初始化器內(nèi)部必須調(diào)用指定初始化器

如果有繼承關(guān)系的類,子類的指定初始化器必須調(diào)用直系父類的指定初始化器

子類的指定初始化器必須調(diào)用直系父類的指定初始化器

總結(jié):
1. 如果是指定初始化器,必須調(diào)用直系父類的指定初始化器;
2. 如果是便捷初始化器,必須調(diào)用自己的指定初始化器;

那便捷初始化器可不可以調(diào)用便捷初始化器呢?

便捷初始化器調(diào)用便捷初始化器

從上圖可以看到,便捷初始化器是可以調(diào)用便捷初始化器的,但是必須遵從一條規(guī)則:便捷初始化器最終必須要調(diào)用指定初始化器

三: 初始化過程

swift 的初始化工作有著一套完善且安全的流程,如果不搞清楚初始化流程,很容易出現(xiàn)各種各樣的錯誤,比如:

初始化錯誤

swift 的初始化過程可以分為兩個階段1:初始化所有屬性,2:正常使用 self 對象

一: 初始化所有屬性

上面已經(jīng)講過,初始化器唯一的原則就是要保證所有屬性都被初始化.我們把從外部調(diào)用初始化器開始到所有屬性初始化完畢這一階段的流程梳理一下:

  1. 外部調(diào)用初始化器(便捷初始化器或者指定初始化器)

  2. 分配堆空間內(nèi)存,此時還未初始化

  3. 指定初始化器必須先確保當(dāng)前類所有的存儲屬性都被初始化
    3.1 如果當(dāng)前類的屬性沒有初始化就調(diào)用父類初始化器,會直接報錯:


  4. 調(diào)用父類指定初始化器,一直向上調(diào)用,直到基類,形成初始化鏈.

二: 正常使用 self 對象

第一階段完成后,所有子類和基類的所有屬性都已經(jīng)初始化了.此時會從基類開始,從上到下依次初始化完成.此時才可以正常使用self

如果第一階段沒有完成,是無法正常使用self的:

也就是說,self的必須在init之后使用

四: 初始化器的重寫
  1. 當(dāng)子類重寫父類的指定初始化器時,必須加上override關(guān)鍵字.子類可以把父類的指定初始化器重寫為指定初始化器和便捷初始化器.

class Person{
    var age: Int
    var name: String
    init(){
        self.age = 0
        self.name = ""
    }
    
    init(age: Int, name: String){
        self.age = age
        self.name = name
    }
    
}

class Student: Person {
    var score: Int
    //重寫父類無參的指定初始化器
    override init() {
        self.score = 0
        super.init()
    }
    //把父類指定初始化器重寫為 便捷初始化器
    override convenience init(age: Int, name: String) {
        self.init()
    }
}


  1. 如果子類寫了一個和父類便捷初始化器一模一樣的初始化器,不用加override,嚴(yán)格來講,這并不是重寫.

因為便捷初始化器是橫向調(diào)用的,它只能在類本身調(diào)用,子類是不能調(diào)用父類的便捷初始化器的.而重寫可以通過super調(diào)用父類的方法,但是卻不能通過super調(diào)用父類的便捷初始化器,這就沖突了.

四: 初始化器的繼承
  1. 如果子類沒有自定義任何指定初始化器,它將繼承父類所有指定初始化器

注意我們我們說的是沒有定義任何指定初始化器,就可以繼承父類的所有指定初始化器.所以即使我們定義了便捷初始化器,仍然會繼承父類所有的指定初始化器:

  1. 如果子類提供了父類所有指定初始化器的實現(xiàn)(兩種方式: 1: 通過第一條繼承 2: 通過重寫),子類會自動繼承所有父類的便捷初始化器
方式一:通過繼承實現(xiàn)了父類所有指定初始化器的實現(xiàn)
方式二:通過重寫父類所有指定初始化器
五: required

required修飾的指定初始化器,注意是指定初始化器.表明所有的子類都必須實現(xiàn)此指定初始化器.并且子類實現(xiàn)的時候不需加override,只需要加上required即可.如果我們要強制子類必須實現(xiàn)某一個初始化器,可以使用required關(guān)鍵字.

required
六: 可失敗初始化器

枚舉,結(jié)構(gòu)體,類都可以用init? 或者 init!定義可失敗初始化器.

可失敗初始化器要注意以下幾點:

  1. 可失敗初始化器可以重寫為非可失敗初始化器;但是非可失敗初始化器不能重寫為可失敗初始化器.這也很容易理解:可失敗(0 和 1)包容非可失敗(1).
可失敗可以重寫為非可失敗
非可失敗不能重寫為可失敗

2.在非可失敗初始化器中調(diào)用可失敗初始化器,要加感嘆號!,但是這樣做不安全

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末毛甲,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌多搀,老刑警劉巖拾弃,帶你破解...
    沈念sama閱讀 210,914評論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡叉庐,警方通過查閱死者的電腦和手機裹匙,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 89,935評論 2 383
  • 文/潘曉璐 我一進(jìn)店門瑞凑,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人概页,你說我怎么就攤上這事籽御。” “怎么了惰匙?”我有些...
    開封第一講書人閱讀 156,531評論 0 345
  • 文/不壞的土叔 我叫張陵技掏,是天一觀的道長。 經(jīng)常有香客問我项鬼,道長哑梳,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,309評論 1 282
  • 正文 為了忘掉前任绘盟,我火速辦了婚禮鸠真,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘龄毡。我一直安慰自己吠卷,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 65,381評論 5 384
  • 文/花漫 我一把揭開白布稚虎。 她就那樣靜靜地躺著撤嫩,像睡著了一般。 火紅的嫁衣襯著肌膚如雪蠢终。 梳的紋絲不亂的頭發(fā)上序攘,一...
    開封第一講書人閱讀 49,730評論 1 289
  • 那天,我揣著相機與錄音寻拂,去河邊找鬼程奠。 笑死,一個胖子當(dāng)著我的面吹牛祭钉,可吹牛的內(nèi)容都是我干的瞄沙。 我是一名探鬼主播,決...
    沈念sama閱讀 38,882評論 3 404
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼距境!你這毒婦竟也來了申尼?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,643評論 0 266
  • 序言:老撾萬榮一對情侶失蹤垫桂,失蹤者是張志新(化名)和其女友劉穎师幕,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體诬滩,經(jīng)...
    沈念sama閱讀 44,095評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡霹粥,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,448評論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了疼鸟。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片后控。...
    茶點故事閱讀 38,566評論 1 339
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖空镜,靈堂內(nèi)的尸體忽然破棺而出浩淘,到底是詐尸還是另有隱情,我是刑警寧澤姑裂,帶...
    沈念sama閱讀 34,253評論 4 328
  • 正文 年R本政府宣布馋袜,位于F島的核電站,受9級特大地震影響舶斧,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜察皇,卻給世界環(huán)境...
    茶點故事閱讀 39,829評論 3 312
  • 文/蒙蒙 一茴厉、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧什荣,春花似錦矾缓、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,715評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至桅锄,卻和暖如春琉雳,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背友瘤。 一陣腳步聲響...
    開封第一講書人閱讀 31,945評論 1 264
  • 我被黑心中介騙來泰國打工翠肘, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人辫秧。 一個月前我還...
    沈念sama閱讀 46,248評論 2 360
  • 正文 我出身青樓束倍,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子绪妹,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 43,440評論 2 348