Objective C 之Runtime訊息機制

2021-07-11 03:11:27 字數 2185 閱讀 5172

最近在找工作,objective-c中的runtime是經常被問到的乙個問題,幾乎是面試大公司必問的乙個問題。當然還有一些其他問題也幾乎必問,例如:runloop,block,記憶體管理等。其他的問題如果有機會我會在其他文章中介紹。本篇文章主要介紹runtime。

runtime簡稱執行時。就是系統在執行的時候的一些機制,其中最主要的是訊息機制。對於c語言,函式的呼叫在編譯的時候會決定呼叫哪個函式( c語言的函式呼叫請看這裡 )。編譯完成之後直接順序執行,無任何二義性。oc的函式呼叫成為訊息傳送。屬於動態呼叫過程。在編譯的時候並不能決定真正呼叫哪個函式(事實證明,在編譯階段,oc可以呼叫任何函式,即使這個函式並未實現,只要申明過就不會報錯。而c語言在編譯階段就會報錯)。只有在真正執行的時候才會根據函式的名稱找到對應的函式來呼叫。

那oc是怎麼實現動態呼叫的呢?下面我們來看看oc通過傳送訊息來達到動態呼叫的秘密。假如在oc中寫了這樣的乙個**:

1

[obj maketext];

其中obj是乙個物件,maketext是乙個函式名稱。對於這樣乙個簡單的呼叫。在編譯時runtime會將上述**轉化成

1

objc_msgsend(obj,@selector(maketext));

首先我們來看看obj這個物件,ios中的obj都繼承於nsobject。

1

2

3

@inte***ce nsobject

在nsobjcet中存在乙個class的isa指標。然後我們看看class:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

typedef struct objc_class *class;

struct objc_class

我們可以看到,對於乙個class類中,存在很多東西,下面我來一一解釋一下:

class isa:指向metaclass,也就是靜態的class。一般乙個obj物件中的isa會指向普通的class,這個class中儲存普通成員變數和物件方法(「-」開頭的方法),普通class中的isa指標指向靜態class,靜態class中儲存static型別成員變數和類方法(「+」開頭的方法)。

class super_class:指向父類,如果這個類是根類,則為null。

注意:所有metaclass中isa指標都指向跟metaclass。而跟metaclass則指向自身。root metaclass是通過繼承root class產生的。與root class結構體成員一致,也就是前面提到的結構。不同的是root metaclass的isa指標指向自身。

class類中其他的成員這裡就先不做過多解釋了,下面我們來看看:

@selector (maketext):這是乙個sel方法選擇器。sel其主要作用是快速的通過方法名字(maketext)查詢到對應方法的函式指標,然後呼叫其函式。sel其本身是乙個int型別的乙個位址,位址中存放著方法的名字。對於乙個類中。每乙個方法對應著乙個sel。所以ios類中不能存在2個名稱相同的方法,即使引數型別不同,因為sel是根據方法名字生成的,相同的方法名稱只能對應乙個sel。

下面我們就來看看具體訊息傳送之後是怎麼來動態查詢對應的方法的。

首先,編譯器將**[obj maketext];轉化為objc_msgsend(obj, @selector (maketext));,在objc_msgsend函式中。首先通過obj的isa指標找到obj對應的class。在class中先去cache中通過sel查詢對應函式method(猜測cache中method列表是以sel為key通過hash表來儲存的,這樣能提高函式查詢速度),若cache中未找到。再去methodlist中查詢,若methodlist中未找到,則取superclass中查詢。若能找到,則將method加入到cache中,以方便下次查詢,並通過method中的函式指標跳轉到對應的函式中去執行。

Objective C語法之Category的使用

無論乙個類設計的如何完美,都不可避免的會遇到沒有 到的需求,那怎麼擴充套件現有的類呢?當然,繼承是個不錯的選擇。但是objective c提供了一種特別的方式來擴充套件類,叫catagory,可以動態的為已經存在的類新增新的行為。這樣可以保證類的原原來的基礎上,較小的改動就可以增加需要的功能。使用c...

Objective C語法之Category的使用

無論乙個類設計的如何完美,都不可避免的會遇到沒有 到的需求,那怎麼擴充套件現有的類呢?當然,繼承是個不錯的選擇。但是objective c提供了一種特別的方式來擴充套件類,叫catagory,可以動態的為已經存在的類新增新的行為。這樣可以保證類的原原來的基礎上,較小的改動就可以增加需要的功能。使用c...

Objective C之Category的使用

category是oc的特有的語法 分類的宣告 inte ce classname categoryname newmethod 在類別中新增方法 不允許在類別中新增變數 end 分類的實現 implementation classname categoryname newmethod end cl...