iOS Swift結構體與類的方法調度

前言

hello,小伙伴們:在忙碌中閑暇之余給大家聊聊swift的知識點,今天給大家帶來的是swift中結構體與類的方法調度詳細區別,希望對你有所幫助,好了廢話不用多說,接下來步入主題!

1.普通方法時兩者方法調度的區別

● 結構體中的普通方法調度是靜態派發的方式
      ○ 詳細分析會在以后: 方法調度之普通結構體方法 闡述
● 類中的普通方法是以函數派發的方式去調度的。
      ○  詳細分析會在以后:方法調度之普通方法 闡述

2.協議中兩者方法調度的區別

● 以類/結構體直接聲明的,
○ 結構體:方法調度都是靜態調度
○ 類:方法調度都是函數調度
● 以協議類型聲明的, 無論協議的實現是類還是結構體:
○ 方法最初定義在協議本身內, 則方法以協議函數表的方式調度
○ 方法最初定義在協議延展內, 則方法以靜態派發的方式調度

3.extension對類中方法調度的影響

swift:
extension PersonClass {
    func changClassName10() {}
}

SIL代碼:


SIL
SIL

斷點,匯編跟蹤一下:


SIL

可以看到 changClassName10 這個方法在執行的時候,由function_ref修飾,sil_vtable 中的函數列表里面沒有。在編譯時已經確定了函數的地址,運行時,直接執行。所以延展內的方法是靜態派發。

?? 思考:為什么普通函數放到了延展中,它就不在函數表中,不是函數派發的方式調度了呢?

我們在方法調度之普通方法一文中講解過:函數表是數組結構,里面的函數是按順序排列的

如果父類存在延展方法,且放在函數表里,就需要考慮它和子類方法的排列順序問題。哪個在前,取決于文件的編譯順序。如果子類先編譯,父類后編譯,還要將子類的所有方法都順次移位,再將延展方法插入到父類方法之后。這樣做,編譯效率就會降低將延展方法使用靜態派發,是一種以空間換時間的方法。協議的延展中的方法,也是靜態派發的,他們是一樣的道理。

【注意】類的延展方法時,需要注意:
● 不可以在子類里重寫父類延展里面的方法,子類可以重寫父類本類定義的方法
● 不可以在延展里 存在/重寫 已在繼承連中存在的同名方法。

swift

4.修飾詞對類方法調度的影響

1. 訪問修飾符修飾的方法

swift
    private func changClassName2() {}
    fileprivate func changClassName3() {}
    public func changClassName4() {}
    internal func changClassName5() {}
    open func changClassName6() {}

SIL 代碼:
sil_vtable SIL :

sil_vtable SIL

雖然所有函數修飾符修飾的方法,都在函數表中存在,但是明顯 private 修飾的 changClassName2 , 與 fileprivate 修飾的changClassName3 與眾不同,他們在方法名的后面有** in _12232F587A4C5CD8B1EEDF696793A4FC **。 這個不同,會導致它們在方法調度的時候,和其他的訪問修飾符什么區別呢?
再看方法調度 SIL :'

SIL

可以發現 private 修飾的 changClassName2 , 與 fileprivate 修飾的changClassName3 在調用時,前面的修飾符是由function_ref 修飾,而不是class_method修飾。所以是靜態派發?
再匯編調試一下:

SIL

編譯時已經確定了函數的地址,運行時,直接執行。所以private/fileprivate 訪問修飾符修飾的是靜態派發。
前面我們提到“函數表存放類中可能是動態派發去執行的函數”, 注意是可能哦, 不是一定的。

小結:
private/fileprivate 訪問修飾符修飾的是靜態派發。
public/open/internal 訪問修飾符修飾的是函數派發。

2. @objc 修飾的方法: 函數表

源碼:

swift
@objc func changClassName7() {}

vtable SIL:

vtable SIL

方法調度 SIL:

vtable SIL

運行、匯編:

匯編

所以: 在swift 中調用 @objc 修飾的方法是函數派發,沒什么特別的。
那 @objc 的作用是什么呢?

我們來看一下changClassName7 方法定義在 SIL 代碼:

匯編

可以看到,除了正常的定義changClassName7 方法以外,額外底層多生成了一個 @objc main.PersonClass.changClassName7()這個方法內部又調用了 正常定義的changClassName7。

所以這個方法是**暴露給OC中調用的接口方法. 沒有@objc 修飾的方法,OC 中是無法使用的。具體的混編步驟,可以參考這篇文章? **Swift 與 OC 混編

3. dynamic 修飾的方法:函數表

源碼如下:

swift
dynamic func changClassName8() {}

vtable SIL:


vtable SIL

方法調度 SIL:

方法調度 SIL

運行、匯編:


匯編

在編譯時,不能確定方法的地址,在函數表內,所以dynamic的方法調度方式是函數派發。

dynamic 有什么作用呢?
看看方法定義SIL:


dynamic

與普通函數不同的是,在方法定義時,多了一個dynamically_replacable的標簽,表明這是一個動態方法,可以被替換。可被替換是指在OC運行時的方法交換的場景下可被替換。
**如果想要對Swift 方法進行方法交換,需要對被替換的方法加dynamic修飾。 **
再使用@_dynamicReplacement(for: teach)來完成替換.
示例代碼如下:

swift
class PersonClass: NSObject {
    dynamic func teach() {
        print("teach")
    }
}

extension PersonClass {
    // swift 5 中提供的方法交換方式
    // 將 teach 方法替換成這行代碼下面的teach1方法
    // 執行 teach 方法,實際上執行的是 teach1方法
    @_dynamicReplacement(for: teach)
    func teach1() {
        print("teach1")
    }
}
let t = PersonClass()
t.teach()

所以打印結果是:“teach1”


teach1

4. @objc dynamic 修飾的方法:消息轉發

源碼如下:

swift
@objc dynamic func changClassName9() {}

vtable SIL:

vtable SIL

函數表中沒有changClassName9的函數。

方法調度 SIL:

方法調度 SIL

與普通的函數派發方法調用時不同,不是以 class_method 方式,是以objc_method方式
運行、匯編調試:

匯編調試

匯編調試時,看到了熟悉的objc_msgSend。這是OC的消息轉發的方式進行方法調度。

5. static 修飾

static修飾的方法,叫做類方法,可以直接由類名去調用,無需創建實例對象。
源碼如下:

swift
static func changClassName11() {}

vtable SIL:


vtable SIL

方法調度 SIL:


方法調度 SIL

運行、匯編調試:


匯編調試

以function_ref 的方式獲取函數, 所以是靜態派發。

6. final 修飾

final修飾符的幾點使用原則
● final修飾符只能修飾類,表明該類不能被其他類繼承,也就是它沒資格當父類。
● final修飾符也可以修飾類中的方法, 表明該方法不能被子類重寫。
● final不能修飾結構體、枚舉、協議。
源碼如下:

swift
final func changClassName1() {}

vtable SIL:


vtable SIL:

方法調度 SIL:

方法調度 SIL

運行、匯編調試:


匯編調試

以function_ref 的方式獲取函數, 所以是靜態派發。

5. 總結

函數表內的函數,不一定是函數派發的方式去調度。但是不在函數表中的,一定不是函數派發的方式。

在調用時獲取函數的方式可以作為判斷調度方法的依據。下面是對應不同的獲取函數的方式的不同調度方式:

Swift 中的方法調度分2大類:動態****調度****與靜態****調度

  • Direct(靜態調度):在 SIL 文件中,以function_ref 的方式獲取函數

  • **?? **結構體的普通方法

  • 類中方法的修飾符為** :**final / private/fileprivate / static

  • 類、結構體、協議延展內的方法

  • **Dynamic Dispatch(****動態調度****)****:****官方文檔傳送門? **Dynamic Dispatch

  • Table(函數表調度) :在 SIL 文件中,以 class_method 的方式,通過 Vtable 獲取函數

  • **?? **普通類中的方法

  • 類中方法的修飾符為:open/public/internal / @objc / dynamaic

  • Message(消息轉發調度):在 SIL 文件中,以 objc_method 的方式獲取函數

  • **?? **@objc dynamaic

  • witness_method(協議表調度):在 SIL 文件中,以 witness_method 的方式, 通過 PWT 獲取函數

  • ?? 遵守了協議并實現了協議本身定義的方法的結構體或者類

好了,小編給大家整理的swift的結構體與類的方法調度,若有收獲,就點個贊吧!

青山不改,綠水長流,后會有期,感謝每一位佳人的支持!

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 227,533評論 6 531
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,055評論 3 414
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 175,365評論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,561評論 1 307
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,346評論 6 404
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 54,889評論 1 321
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 42,978評論 3 439
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,118評論 0 286
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,637評論 1 333
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,558評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,739評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,246評論 5 355
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 43,980評論 3 346
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,362評論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,619評論 1 280
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,347評論 3 390
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,702評論 2 370

推薦閱讀更多精彩內容