Swift之通過減少動態(tài)派發(fā)來提升性能


Swift是OO(面向?qū)ο螅┑恼Z言,所以少不了方法和屬性的重載等特性,程序只能在運行時來確定具體的方法或?qū)傩詠黹g接調(diào)用或間接訪問,這就叫做動態(tài)派發(fā)。從性能上考慮,對于動態(tài)派發(fā)的方法,會有常量時間的運行時開銷。接下來將介紹三種方法來移除這樣的動態(tài)性,final,private,全模塊優(yōu)化(Whole Module Optimization),以此提升性能。

考慮下面的例子:

class ParticleModel {
    var point = ( 0.0, 0.0 )
    var velocity = 100.0

    func updatePoint(newPoint: (Double, Double), newVelocity: Double) {
        point = newPoint
        velocity = newVelocity
    }

    func update(newP: (Double, Double), newV: Double) {
        updatePoint(newP, newVelocity: newV)
    }
}

var p = ParticleModel()
for i in stride(from: 0.0, through: 360, by: 1.0) {
    p.update((i * sin(i), i), newV:i*1000)
}

如上述代碼所示,調(diào)用過程為:

  1. 調(diào)用變量pupdate方法。
  2. 調(diào)用pupdatePoint方法。
  3. 獲取p的元組類型變量point
  4. 獲取p的屬性velocity。

由于ParticleModel可以被子類,所以其方法和屬性就能被重載,這就不可避免的需要使用動態(tài)調(diào)用。

在Swift中,動態(tài)調(diào)用是通過在一個方法表中找到方法然后執(zhí)行間接的調(diào)用(類似于C++的虛函數(shù)表),對于這種先查找再調(diào)用的過程,其效率是要低于方法的直接調(diào)用,而且間接調(diào)用會阻止許多編譯器優(yōu)化,這將加重間接調(diào)用的開銷。接下來將列舉一些技巧來禁用動態(tài)派發(fā)的行為,以達(dá)到提升性能的目的。

當(dāng)屬性、方法、或類不需要被重載時,可在其聲明的地方加上final關(guān)鍵字

在屬性,方法或類聲明時加上final關(guān)鍵字,表示其不能被重載,這將允許編譯器安全的移除動態(tài)派發(fā)。如下代碼所示,pointvelocity將直接從對象的存儲屬性中加載,updatePoint()方法將被直接調(diào)用;另外,update()依然會通過動態(tài)派發(fā)的方式來調(diào)用,這樣,ParticleModel的子類就可以重載update()來自定義實現(xiàn)。

class ParticleModel {
    final var point = ( x: 0.0, y: 0.0 )
    final var velocity = 100.0

    final func updatePoint(newPoint: (Double, Double), newVelocity: Double) {
        point = newPoint
        velocity = newVelocity
    }

    func update(newP: (Double, Double), newV: Double) {
        updatePoint(newP, newVelocity: newV)
    }
}

除了上面所示,在屬性和方法聲明前加final關(guān)鍵字,還可以直接在類上加final,表示該類將不能作為父類被子類化,隱含的表明該類的所有的方法和屬性都是final的。

final class ParticleModel {
    var point = ( x: 0.0, y: 0.0 )
    var velocity = 100.0
    // ...
}

在屬性、方法、或類聲明前加private關(guān)鍵字,將限制其只能在同一個文件中被引用

在聲明前加private關(guān)鍵字,將限制其只能在當(dāng)前文件中被引用,這將允許編譯器在當(dāng)前文件中找到所有潛在的重載聲明,編譯器會對這些private關(guān)鍵字的方法或?qū)傩赃M(jìn)行優(yōu)化,移除間接的方法調(diào)用以及屬性訪問。

假設(shè)在當(dāng)前文件中沒有類重載ParticleModel,那么編譯器將移除所有帶有private聲明的動態(tài)派發(fā)調(diào)用。

class ParticleModel {
    private var point = ( x: 0.0, y: 0.0 )
    private var velocity = 100.0

    private func updatePoint(newPoint: (Double, Double), newVelocity: Double) {
        point = newPoint
        velocity = newVelocity
    }

    func update(newP: (Double, Double), newV: Double) {
        updatePoint(newP, newVelocity: newV)
    }
}

如上代碼所示,pointvelocity將直接訪問,updatePoint()方法也將直接被調(diào)用,而update()方法由于沒有加private關(guān)鍵字,依然是只能間接調(diào)用。
同樣,private可以加在類的聲明前,等同于類的所有方法和屬性都將加上private關(guān)鍵字。

private class ParticleModel {
    var point = ( x: 0.0, y: 0.0 )
    var velocity = 100.0
    // ...
}

在使用internal的聲明中通過使用Whole Module Optimization來隱式的推斷出final

默認(rèn)的情況下,Xcode將單獨編譯源文件,這會限制編譯器優(yōu)化的程度,Xcode 7后,增加了Whole Module Optimization選項,它能允許編譯器在同一個模塊(Module)中分析所有的源文件來進(jìn)行優(yōu)化,可以在Xcode的Building Settings中開啟該選項,如下圖所示。

在開啟Whole Module Optimization選項,且聲明為internal(默認(rèn)級別)的情況下,模塊的所有文件將同時被編譯,這將允許編譯器對整個模塊一起分析,并對沒有被重載且聲明為internal級別的類、方法或?qū)傩蕴砑?code>final關(guān)鍵字。
如下代碼所示,我們修改一下ParticleModel類,添加public關(guān)鍵字:

public class ParticleModel {
   var point = ( x: 0.0, y: 0.0 )
   var velocity = 100.0

   func updatePoint(newPoint: (Double, Double), newVelocity: Double) {
       point = newPoint
       velocity = newVelocity
   }

   public func update(newP: (Double, Double), newV: Double) {
       updatePoint(newP, newVelocity: newV)
   }
}

var p = ParticleModel()
for i in stride(from: 0.0, through: times, by: 1.0) {
   p.update((i * sin(i), i), newV:i*1000)
}

如上代碼,當(dāng)開啟Whole Module Optimization選項的情況下,編譯器能在屬性point,velotity,以及updatePoint()方法上推斷出final,既相當(dāng)于在pointvelocity、updatePoint()聲明前加上final關(guān)鍵字,而update()方法由于是public級別,所以無法推斷出final關(guān)鍵字,其仍將是間接調(diào)用。

總結(jié):

  • 當(dāng)使用privatefinal關(guān)鍵字,或者在開啟Whole Module Optimization選項,聲明為internal級別的沒有被重載的方法下,將直接調(diào)用,在編譯時確定。
  • 運行時決定的動態(tài)派發(fā)的情形包括:
    • 繼承自NSObject或者方法有@objc前綴。
    • 使用Swift的方法表的方式,除去上述情況下,將采用這種方式。
最后編輯于
?著作權(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)容

  • 1. Java基礎(chǔ)部分 基礎(chǔ)部分的順序:基本語法,類相關(guān)的語法,內(nèi)部類的語法,繼承相關(guān)的語法,異常的語法,線程的語...
    子非魚_t_閱讀 34,626評論 18 399
  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn),斷路器,智...
    卡卡羅2017閱讀 136,506評論 19 139
  • 對象的創(chuàng)建與銷毀 Item 1: 使用static工廠方法,而不是構(gòu)造函數(shù)創(chuàng)建對象:僅僅是創(chuàng)建對象的方法,并非Fa...
    孫小磊閱讀 2,177評論 0 3
  • 前言 人生苦多,快來 Kotlin ,快速學(xué)習(xí)Kotlin! 什么是Kotlin? Kotlin 是種靜態(tài)類型編程...
    任半生囂狂閱讀 26,669評論 9 118
  • 估計最近用眼過多,白天上班看電腦,晚上還要盯手機,雙眼又脹又癢,甚至引起了頭昏頭疼。就這樣還要參與各類年終聚會,我...
    磚兒zr閱讀 316評論 16 13

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