Swift 派發(fā)機(jī)制

此篇博客用來自我學(xué)習(xí),來源戴銘大佬的這篇博客

Swift 派發(fā)機(jī)制

派發(fā)目的是讓 CPU 知道被調(diào)用的函數(shù)在哪里。Swift 語言是支持編譯型語言的直接派發(fā),函數(shù)表派發(fā)和消息機(jī)制派發(fā)三種派發(fā)方式的,下面分別對這三種派發(fā)方式說明下。

直接派發(fā)

C++ 默認(rèn)使用的是直接派發(fā),加上 virtual修飾符可以改成函數(shù)表派發(fā)。直接派發(fā)是最快的,原因是調(diào)用指令會少,還可以通過編譯器進(jìn)行比如內(nèi)聯(lián)等方式的優(yōu)化。缺點是由于缺少動態(tài)性而不支持繼承。

struct DragonFirePosition {
    var x:Int64
    var y:Int32
    func land() {}
}
func DragonWillFire(_ position:DragonFirePosition) {
    position.land()
}
let position = DragonFirePosition(x: 342, y: 213)
DragonWillFire(position)

編譯 inlineDragonWillFire(DragonFirePosition(x: 342, y: 213))會直接跳到方法實現(xiàn)的地方,結(jié)果就變成 position.land()。

函數(shù)表派發(fā)

Java 默認(rèn)就是使用的函數(shù)表派發(fā),通過 final 修飾符改成直接派發(fā)。函數(shù)表派發(fā)是有動態(tài)性的,在 Swift 里函數(shù)表叫 witness table,大部分語言叫 virtual table。一個類里會用數(shù)組來存儲里面的函數(shù)指針,override 父類的函數(shù)會替代以前的函數(shù),子類添加的函數(shù)會被加到這個數(shù)組里。舉個例子:

class Fish {
    func swim() {}
    func eat() {
        //normal eat
    }
}
class FlyingFish: Fish {
    override func eat() {
        //flying fish eat
    }
    func fly() {}
}

編譯器會給 Fish 類和 FlyingFish 類分別創(chuàng)建 witness table。在 Fish 的函數(shù)表里有 swimeat 函數(shù),在 FlyingFish 函數(shù)表里有父類 Fishswim,覆蓋了父類的 eat 和新增加的函數(shù) fly。

一個函數(shù)被調(diào)用時會先去讀取對象的函數(shù)表,再根據(jù)類的地址加上該的函數(shù)的偏移量得到函數(shù)地址,然后跳到那個地址上去。從編譯后的字節(jié)碼這方面來看就是兩次讀取一次跳轉(zhuǎn),比直接派發(fā)還是慢了些。

消息機(jī)制派發(fā)

這種機(jī)制是在運(yùn)行時可以改變函數(shù)的行為,KVOCoreData 都是這種機(jī)制的運(yùn)用。OC 默認(rèn)就是使用的消息機(jī)制派發(fā),使用 C 來直接派發(fā)獲取高性能。Swift 可以通過 dynamic 修飾來支持消息機(jī)制派發(fā)。

當(dāng)一個消息被派發(fā),運(yùn)行時就會按照繼承關(guān)系向上查找被調(diào)用的函數(shù)。但是這樣效率不高,所以需要通過緩存來提高效率,這樣查找性能就能和函數(shù)派發(fā)差不多了。

具體派發(fā)

聲明

值類型都會采用直接派發(fā)。無論是 class 還是協(xié)議extension 也都是直接派發(fā)。class協(xié)議是函數(shù)表派發(fā)。

指定派發(fā)方式

  • final:讓類里的函數(shù)使用直接派發(fā),這樣該函數(shù)將會沒有動態(tài)性,運(yùn)行時也沒法取到這個函數(shù)。
  • dynamic:可以讓類里的函數(shù)使用消息機(jī)制派發(fā),可以讓 extension 里的函數(shù)被 override。

派發(fā)優(yōu)化

Swift 會在這上面做優(yōu)化,比如一個函數(shù)沒有 override,Swift 就可能會使用直接派發(fā)的方式,所以如果屬性綁定了 KVO 它的 getter和 setter 方法可能會被優(yōu)化成直接派發(fā)而導(dǎo)致 KVO 的失效,所以記得加上 dynamic 的修飾來保證有效。后面 Swift 應(yīng)該會在這個優(yōu)化上去做更多的處理。

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

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

  • 原文: Method Dispatch in Swift作者: Brain King譯者: kemchenj 譯者...
    kemchenj閱讀 9,547評論 11 91
  • Swift之所以速度比Object-c快,我覺得跟他的派發(fā)機(jī)制有關(guān)。下面我們聊聊Swift的派發(fā)機(jī)制。 派發(fā)機(jī)制分...
    xgou閱讀 1,405評論 0 2
  • 時間過得真快,二寶已經(jīng)一歲四個月了,小腿很溜。家里的桌子,柜子,窗臺,已經(jīng)成為他攀爬的目標(biāo)。小胖腿努力向上...
    孔雀堂茶號閱讀 558評論 0 3
  • 1 終于讀完<<紅樓夢>>,在此之前,我一直認(rèn)為這是一個不可能完成的任務(wù)。只是當(dāng)我拿起這本書后,就再也舍不得放下。...
    邵曉寧閱讀 739評論 6 4
  • 環(huán)境: 系統(tǒng):centos7ps: 請確認(rèn)kafka,zookeeper,storm部署完成(本文基于Apache...
    jason_李閱讀 2,310評論 0 1

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