Swift結(jié)構(gòu)體與類的方法調(diào)度

前言

在忙碌中抽取時(shí)間來看這篇文章的朋友,希望各位看完這篇問斬都能都有所收獲。這篇文章主要是給大家分享的是Swift結(jié)構(gòu)體與類的方法調(diào)度。好了廢話不多說,接下來直接步入主題吧。

1.普通方法時(shí)兩者方法調(diào)度的區(qū)別

● 結(jié)構(gòu)體中的普通方法調(diào)度是靜態(tài)派發(fā)的方式
   ○ 詳細(xì)分析會在以后: 方法調(diào)度之普通結(jié)構(gòu)體方法 闡述
● 類中的普通方法是以函數(shù)派發(fā)的方式去調(diào)度的。
   ○ 詳細(xì)分析會在以后:方法調(diào)度之普通方法 闡述

2.議中兩者方法調(diào)度的區(qū)別

● 以類/結(jié)構(gòu)體直接聲明的,
  ○ 結(jié)構(gòu)體:方法調(diào)度都是靜態(tài)調(diào)度
  ○ 類:方法調(diào)度都是函數(shù)調(diào)度
● 以協(xié)議類型聲明的, 無論協(xié)議的實(shí)現(xiàn)是類還是結(jié)構(gòu)體:
  ○ 方法最初定義在協(xié)議本身內(nèi), 則方法以協(xié)議函數(shù)表的方式調(diào)度
  ○ 方法最初定義在協(xié)議延展內(nèi), 則方法以靜態(tài)派發(fā)的方式調(diào)度

3.extension對類中方法調(diào)度的影響

``extension PersonClass {

``func changClassName10``() {}

}

SIL代碼:

[圖片上傳失敗...(image-1dcbb3-1634395436453)]

[圖片上傳失敗...(image-a6e24-1634395436453)]

斷點(diǎn),匯編跟蹤一下:

[圖片上傳失敗...(image-50578a-1634395436453)]

可以看到 changClassName10 這個(gè)方法在執(zhí)行的時(shí)候,由function_ref修飾,sil_vtable 中的函數(shù)列表里面沒有。在編譯時(shí)已經(jīng)確定了函數(shù)的地址,運(yùn)行時(shí),直接執(zhí)行。所以延展內(nèi)的方法是靜態(tài)派發(fā)。

??****思考: 為什么普通函數(shù)放到了延展中,它就不在函數(shù)表中,不是函數(shù)派發(fā)的方式調(diào)度了呢?

我們在[方法調(diào)度之普通方法:后續(xù)筆者會發(fā)布普通方法調(diào)用過程]:函數(shù)表是數(shù)組結(jié)構(gòu),里面的函數(shù)是按順序排列的

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

【注意】類的延展方法時(shí),需要注意:

  • 不可以在子類里重寫父類延展里面的方法,子類可以重寫父類本類定義的方法
  • 不可以在延展里 存在/重寫 已在繼承連中存在的同名方法

[圖片上傳失敗...(image-5410e6-1634395436453)]

4.修飾詞對類方法調(diào)度的影響

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

private func changClassName2``() {}

fileprivate func changClassName3``() {}

public func changClassName4``() {}

internal func changClassName5``() {}

open func changClassName6``() {}

SIL 代碼:
sil_vtable SIL :

[圖片上傳失敗...(image-5ddce8-1634395436453)]
雖然所有函數(shù)修飾符修飾的方法,都在函數(shù)表中存在,但是明顯 private 修飾的 changClassName2 , 與 fileprivate 修飾的changClassName3 與眾不同,他們在方法名的后面有 in _12232F587A4C5CD8B1EEDF696793A4FC ****。 這個(gè)不同,會導(dǎo)致它們在方法調(diào)度的時(shí)候,和其他的訪問修飾符什么區(qū)別呢?
再看方法調(diào)度 SIL :

[圖片上傳失敗...(image-b50363-1634395436453)]
可以發(fā)現(xiàn) private 修飾的 changClassName2 , 與 fileprivate 修飾的changClassName3 在調(diào)用時(shí),前面的修飾符是由function_ref 修飾 而不是class_method修飾。所以是靜態(tài)派發(fā)?

再匯編調(diào)試一下:

[圖片上傳失敗...(image-8148de-1634395436453)]

編譯時(shí)已經(jīng)確定了函數(shù)的地址,運(yùn)行時(shí),直接執(zhí)行。所以private/fileprivate 訪問修飾符修飾的是靜態(tài)派發(fā)。

前面我們提到“函數(shù)表存放類中可能是動態(tài)派發(fā)去執(zhí)行的函數(shù)”, 注意是可能哦, 不是一定的

小結(jié):

private/fileprivate 訪問修飾符修飾的是靜態(tài)派發(fā)。

public/open/internal 訪問修飾符修飾的是函數(shù)派發(fā)。

2. @objc 修飾的方法: 函數(shù)表

源碼:@objc func changClassName7``() {}
vtable SIL:

[圖片上傳失敗...(image-d6971f-1634395436453)]
方法調(diào)度 SIL:

[圖片上傳失敗...(image-cdcda3-1634395436453)]
運(yùn)行、匯編:

[圖片上傳失敗...(image-a031ef-1634395436453)]
所以: 在swift 中調(diào)用 @objc 修飾的方法是函數(shù)派發(fā),沒什么特別的。

@objc 的作用是什么呢?

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

[圖片上傳失敗...(image-b852b5-1634395436453)]
可以看到,除了正常的定義changClassName7 方法以外,額外底層多生成了一個(gè) @objc main.PersonClass.changClassName7()這個(gè)方法內(nèi)部又調(diào)用了 正常定義的changClassName7。

所以這個(gè)方法是暴露給OC中調(diào)用的接口方法. 沒有@objc 修飾的方法,OC 中是無法使用的。

3. dynamic 修飾的方法:函數(shù)表

源碼如下:
dynamic func changClassName8``() {}
vtable SIL:

[圖片上傳失敗...(image-f62e21-1634395436453)]
方法調(diào)度 SIL:

[圖片上傳失敗...(image-7b8f92-1634395436453)]
運(yùn)行、匯編:

[圖片上傳失敗...(image-3bbd4f-1634395436453)]
在編譯時(shí),不能確定方法的地址,在函數(shù)表內(nèi),所以 dynamic的方法調(diào)度方式 是函數(shù)派發(fā)。

dynamic 有什么作用呢?

看看方法定義SIL:

[圖片上傳失敗...(image-b25c4b-1634395436453)]
與普通函數(shù)不同的是,在方法定義時(shí),多了一個(gè)dynamically_replacable的標(biāo)簽,表明這是一個(gè)動態(tài)方法,可以被替換。可被替換是指在OC運(yùn)行時(shí)的方法交換的場景下可被替換。

如果想要對Swift 方法進(jìn)行方法交換,需要對被替換的方法加dynamic修飾。

再使用@_dynamicReplacement(for: teach)來完成替換.

示例代碼如下:

````classPersonClass:NSObject{`

``dynamic func teach``() {

``print``(``"teach"``)

``}

}

extension PersonClass {

``// swift 5 中提供的方法交換方式

``// 將 teach 方法替換成這行代碼下面的teach1方法

``// 執(zhí)行 teach 方法,實(shí)際上執(zhí)行的是 teach1方法

``@``_dynamicReplacement``(``for``: ``teach``)

``func teach1``() {

``print``(``"teach1"``)

``}

}

let t = ``PersonClass``()

t.teach()

所以打印結(jié)果是:“teach1”

[圖片上傳失敗...(image-f74690-1634395436453)]

4. @objc dynamic 修飾的方法:消息轉(zhuǎn)發(fā)

源碼如下:

1 @objc dynamic func changClassName9``() {}

vtable SIL:

[圖片上傳失敗...(image-9721e7-1634395436453)]

函數(shù)表中沒有changClassName9的函數(shù)。

方法調(diào)度 SIL:

[圖片上傳失敗...(image-5abe6-1634395436453)]
與普通的函數(shù)派發(fā)方法調(diào)用時(shí)不同,不是以 class_method 方式,是以objc_method方式

運(yùn)行、匯編調(diào)試:

[圖片上傳失敗...(image-5b1001-1634395436453)]
匯編調(diào)試時(shí),看到了熟悉的objc_msgSend 。這是OC的消息轉(zhuǎn)發(fā)的方式進(jìn)行方法調(diào)度。

5. static 修飾

static修飾的方法,叫做類方法,可以直接由類名去調(diào)用,無需創(chuàng)建實(shí)例對象。

源碼如下:

1 static func changClassName11``() {}

vtable SIL:

[圖片上傳失敗...(image-21da64-1634395436453)]

方法調(diào)度 SIL:

[圖片上傳失敗...(image-8a3771-1634395436453)]

運(yùn)行、匯編調(diào)試:

[圖片上傳失敗...(image-34e219-1634395436453)]
以function_ref 的方式獲取函數(shù), 所以是靜態(tài)派發(fā)

6. final 修飾

final修飾符的幾點(diǎn)使用原則

  • final修飾符只能修飾,表明該類不能被其他類繼承,也就是它沒資格當(dāng)父類。
  • final修飾符也可以修飾類中的方法, 表明該方法不能被子類重寫
  • final不能修飾結(jié)構(gòu)體、枚舉、協(xié)議。

源碼如下:
源碼如下:

1 final func changClassName1``() {}

table SIL:

[圖片上傳失敗...(image-f3ceea-1634395436453)]

方法調(diào)度 SIL:

[圖片上傳失敗...(image-d02d0a-1634395436453)]

行、匯編調(diào)試:

[圖片上傳失敗...(image-807e0b-1634395436453)]

以function_ref 的方式獲取函數(shù), 所以是靜態(tài)派發(fā)

5.總結(jié)

函數(shù)表內(nèi)的函數(shù),不一定是函數(shù)派發(fā)的方式去調(diào)度。但是不在函數(shù)表中的,一定不是函數(shù)派發(fā)的方式。

在調(diào)用時(shí)獲取函數(shù)的方式可以作為判斷調(diào)度方法的依據(jù)。下面是對應(yīng)不同的獲取函數(shù)的方式的不同調(diào)度方式:

Swift 中的方法調(diào)度分2大類:動態(tài) 調(diào)度 與靜態(tài) 調(diào)度

  • Direct(靜態(tài)調(diào)度): 在 SIL 文件中,以function_ref 的方式獲取函數(shù)
    • ?? 結(jié)構(gòu)體的普通方法
    • 類中方法的修飾符為 final / private/fileprivate / static
    • 類、結(jié)構(gòu)體、協(xié)議延展內(nèi)的方法
  • Dynamic Dispatch( 動態(tài)調(diào)度 官方文檔傳送門? Dynamic Dispatch
    • Table(函數(shù)表調(diào)度) : 在 SIL 文件中,以 class_method 的方式,通過 Vtable 獲取函數(shù)
    • ?? 普通類中的方法
    • 類中方法的修飾符為: open/public/internal / @objc / dynamaic
    • Message(消息轉(zhuǎn)發(fā)調(diào)度): 在 SIL 文件中,以 objc_method 的方式獲取函數(shù)
    • ?? @objc dynamaic
    • witness_method(協(xié)議表調(diào)度): 在 SIL 文件中,以 witness_method 的方式, 通過 PWT 獲取函數(shù)
    • ?? 遵守了協(xié)議并實(shí)現(xiàn)了協(xié)議本身定義的方法的結(jié)構(gòu)體或者類

好了,小編給大家整理的swift的結(jié)構(gòu)體與類的方法調(diào)度,若有收獲,就點(diǎn)個(gè)贊吧!

青山不改,綠水長流,后會有期,感謝每一位佳人的支持!更多iOS知識請關(guān)注主頁。

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

推薦閱讀更多精彩內(nèi)容