幾個有關iOS的幾個常見問題-----RunTime

來源:互聯網
上載者:User

幾個有關iOS的幾個常見問題-----RunTime

說起RunTime這又是一個痛點,學iOS開發有多少是純粹的因為興趣?然後又因為興趣去研究OC。畢竟大學裡買的起iPhone,買的起MacBook,買的起開發人員帳號(xCode7以前如果真機測試是需要開發人員帳號的)?所以我在這裡建議那些面試官,別問你在大學裡如何學習的iOS開發的。。。。因為大多數都是像我這樣有人帶出來的,自學Java遇到不懂的你可以在學校問老師,iOS你丫問誰去?最重要的是iOS火起來是從12年開始的。。。12年你開始幹,而且環視國內沒啥教程的幹。。。10個程式員有3個能學到考慮Runtime這個點就真的很不錯了。。。。。所以問這個個人感覺真的沒太必要。。。。而且第一批iOS開發人員,都是有了四五年開發經驗後才開始的iOS開發,你們覺得呢?哪種人真的很少的。。。。好了,吐槽結束。。。。開始我們的重點------Runtime

RunTime

字面上看就是“啟動並執行時候”,說得稍微專業點就是程式在運行時的狀態(我在這兒說一下,Runtime是OC語言的專業知識,不是iOS開發的專業知識,而且Java也有Runtime的)。

一、動態 vs 靜態語言

OC 是面相運行時的語言,意思是它會儘可能的把編譯和連結時要執行的邏輯延遲到運行時。這就給了程式員很大的靈活性,程式員可以根據需要把訊息定向給合適的對象,你甚至可以用交換方法的實現(在OC 中調用一個對象的方法可以看成向一個對象發送訊息)。這就需要使用 runtime,runtime 可以做對象自省查看他們正在做的和不能做的,並且合適的分發訊息(譯註:感興趣的同學可以查看 NSObject 類的 – forwardingTargetForSelector: 和 – forwardInvocation: 方法。P.S. 不是 NSObject 協議! )。如果我們和 C 這樣的語言對比。在 C 裡,你從 main() 方法開始寫然後就是從上到下的寫邏輯了並按你寫代碼的順序執行程式。一個 C 的結構體不能轉寄函數執行請求到其他的目標上(other targets)。很可能你的程式是這樣的:

 



int main(int argc, const char **argv[]){            printf("Hello World!");    return 0;}
編譯器解析,最佳化然後把最佳化後的代碼轉成彙編:
.text .align 4,0x90 .globl _main_main:Leh_func_begin1: pushq %rbpLlabel1: movq %rsp, %rbpLlabel2: subq $16, %rspLlabel3: movq %rsi, %rax movl %edi, %ecx  movl %ecx, -8(%rbp) movq %rax, -16(%rbp)  xorb %al, %al  leaq LC(%rip), %rcx  movq %rcx, %rdi  call _printf  movl $0, -4(%rbp)  movl -4(%rbp), %eax  addq $16, %rsp  popq %rbp  retLeh_func_end1:  .cstringLC:  .asciz "Hello World!"
然後連結庫並產生可執行程式要和 OC 對比的話,處理過程很相似,產生的程式碼依賴於是否有 OC Runtime 庫。當剛學 OC 時,我們最先瞭解的(最簡單的那種)是 OC 中用括弧包起來的代碼像這樣…
[self doSomethingWithVar:var1];
再然後:
objc_msgSend(self,@selector(doSomethingWithVar:),var1);
之後的麼。。。我就找不到是啥了。。。。

二、OC Runtime 是什嗎?

OC 的 Runtime 是一個執行階段程式庫(Runtime Library),它是一個主要使用 C 和彙編寫的庫,為 C 添加了面相對象的能力並創造了 OC。這就是說它在類資訊(Class information) 中被載入,完成所有的方法分發,方法轉寄,等等。OC runtime 建立了所有需要的結構體,讓 OC 的面相對象編程變為可能。

三、OC Runtime 術語

更深入之前,咱們先瞭解點術語。Mac 和 iPhone 開發人員關心的有兩個 runtime:Modern Runtime(現代的 Runtime) 和 Legacy Runtime(過時的 Runtime)。Modern Runtime:覆蓋所有 64 位元的 Mac OS X 應用和所有 iPhone OS 的應用。 Legacy Runtime: 覆蓋其他的所有應用(所有 32 位的 Mac OS X 應用) Method 有 2 種基本類型的方法。Instance Method(執行個體方法):以 ‘-’ 開始,比如 -(void)doFoo; 在對象執行個體上操作。Class Method(類方法):以 ‘+’ 開始,比如 +(id)alloc。方法(Methods)和 C 的函數很像,是一組代碼,執行一個小的任務,如:


- (NSString *)movieTitle { return @"Futurama: Into the Wild Green Yonder"; }
Selector 在 Objective-C 中 selector 只是一個 C 的資料結構,用於表示一個你想在一個對象上執行的 OC 方法。在 runtime 中的定義像這樣…

typedef struct objc_selector  *SEL;
像這樣使用…
SEL aSel = @selector(movieTitle);
Message(訊息)
[target getMovieTitleForObject:obj];
訊息是方括弧 ‘[]’ 中的那部分,由你要向其發送訊息的對象(target),你想要在上面執行的方法(method)還有你發送的參數(arguments)組成。 OC 的訊息和 C 函數調用是不同的。事實上,你向一個對象發送訊息並不意味著它會執行它。Object(對象)會檢查訊息的寄件者,基於這點再決定是執行一個不同的方法還是轉寄訊息到另一個目標對象上。Class 如果你查看一個類的runtime資訊,你會看到這個…
typedef struct objc_class *Class;typedef struct objc_object {    Class isa;} *id;
這裡有幾個事情。我們有一個 OC 類的結構體和一個對象的結構體。objc_object 只有一個指向類的 isa 指標,就是我們說的術語 “isa pointer”(isa 指標)。這個 isa 指標是當你向對象發送訊息時,OC Runtime 檢查一個對象並且查看它的類是什麼然後開始查看它是否響應這些 selectors 所需要的一切。最後我麼看到了 id 指標。預設情況下 id 指標除了告訴我們它們是 OC 對象外沒有其他用了。當你有一個 id 指標,然後你就可以問這個對象是什麼類的,看看它是否響應一個方法,等等,然後你就可以在知道這個指標指向的是什麼對象後執行更多的操作了。你可以在 LLVM/Clang 的文檔中的 Block 中看到
struct Block_literal_1 {    void *isa; // initialized to &_NSConcreteStackBlock or &_NSConcreteGlobalBlock        int flags;        int reserved;         void (*invoke)(void *, ...);     struct Block_descriptor_1 {         unsigned long int reserved; // NULL             unsigned long int size;  // sizeof(struct Block_literal_1)        // optional helper functions             void (*copy_helper)(void *dst, void *src);        void (*dispose_helper)(void *src);         } *descriptor;        // imported variables};
Blocks 被設計為相容 OC 的 runtime,所以他們被作為對象對待,因此他們可以響應訊息,比如 -retain,-release,-copy ,等等。

IMP(方法實現 Method Implementations)

typedef id (*IMP)(id self,SEL _cmd,...);
IMP 是指向方法實現的函數指標,由編譯器為你產生。如果你新接觸 OC 你現在不需要直接接觸這些,但是我們將會看到,OC runtime 將如何調用你的方法的。OC Classes(OC 類) 那麼什麼是 OC 類?在 OC 中的一個類實現看起來像這樣:

@interface MyClass : NSObject { // vars NSInteger counter; } // methods -(void)doFoo; @end

但是 runtime 不只要追蹤這些
#if !__OBJC2__       Class super_class                        OBJC2_UNAVAILABLE;    const char *name                         OBJC2_UNAVAILABLE;    long version                             OBJC2_UNAVAILABLE;    long info                                OBJC2_UNAVAILABLE;    long instance_size                       OBJC2_UNAVAILABLE;        struct objc_ivar_list *ivars             OBJC2_UNAVAILABLE;        struct objc_method_list **methodLists    OBJC2_UNAVAILABLE;        struct objc_cache *cache                 OBJC2_UNAVAILABLE;        struct objc_protocol_*protocols          OBJC2_UNAVAILABLE;#endif

我們可以看到,一個類有其父類的引用,它的名字,執行個體變數,方法,緩衝還有它遵循的協議。runtime 在響應類或執行個體的方法時需要這些資訊。

四、那麼 Class 定義的是對象還是對象本身?它是如何?的(譯註:讀者需要區分 Class 和 class 是不同的,正如 Nil 和 nil 的用途是不同的)

是的,之前我說過 OC 類也是對象,runtime 通過建立 Meta Classes 來處理這些。當你發送一個訊息像這樣 [NSObject alloc] 你正在向類對象發送一個訊息,這個類對象需要是 MetaClass 的執行個體,MetaClass 也是 root meta class 的執行個體。當你說繼承自 NSObject 時,你的類指向 NSObject 作為自己的 superclass。然而,所有的 meta class 指向 root metaclass 作為自己的 superclass。所有的 meta class 只是簡單的有一個自己響應的方法列表。所以當你向一個類對象發送訊息如 [NSObject alloc],然後實際上 objc_msgSend() 會檢查 meta class 看看它是否響應這個方法,如果他找到了一個方法,就在這個 Class 對象上執行(譯註:class 是一個執行個體對象的類型,Class 是一個類(class)的類型。對於完全的 OO 來說,類也是個對象,類是類類型(MetaClass)的執行個體,所以類的類型描述就是 meta class)。

五、為什麼我們繼承自蘋果的類

從你開始 Cocoa 開發時,那些教程就說如繼承自 NSObject 然後開始寫一些代碼,你享受了很多繼承自蘋果的類所帶來的便利。有一件事你從未意識到的是你的對象被設定為使用 OC 的 runtime。當我們為我們的類的一個執行個體分配了記憶體,像這樣…


MyObject *object = [[MyObject alloc] init];
最先執行的訊息是 +alloc。如果你查看下文檔, 它說“新的執行個體對象的 isa 執行個體變數被初始化為指向一個資料結構,那個資料結構描述了這個類;其他的執行個體變數被初始化為 0。”所以繼承自蘋果的類不僅僅是繼承了一些重要的屬性,也繼承了能在記憶體中輕鬆分配記憶體的能力和在記憶體中建立滿足 runtime 期望的對象結構(設定 isa 指標指向我們的類)。

六、那麼 Class Cache 是什嗎?(objc_cache *cache)

 

當 OC runtime 沿著一個對象的 isa 指標檢查時,它會發現一個對象實現了許多的方法。然而你可能只調用其中一小部分的方法,也沒有意義每次檢查時搜尋這個類的分發表(dispatch table)中的所有 selector。所以這個類實現了一個緩衝,當你搜尋一個類的分發表,並找到合適的 selector 後,就會把它放進緩衝中。所以當 objc_msgSend() 在一個類中尋找 selector 時會先尋找類緩衝。有個理論是,當你在一個類上調用了一個訊息,你很可能之後還會調用它。所以如果我們考慮到這點,就意味著當我們有個子類繼承自 NSObject 叫做 MyObject 並且運行了以下的代碼:

 

MyObject *obj = [[MyObject alloc] init]; @implementation MyObject- (id)init {    if(self = [super init]) {        [self setVarA:@”blah”];        }    return self;}@end

(1) [MyObject alloc] 首先被執行。MyObject 沒有實現 alloc 方法,所以我們不能在這個類中找到 +alloc 方法,然後沿著 superclass 指標會指向 NSObject。

(2) 我們詢問 NSObject 是否響應 +alloc 方法,它可以。+alloc 檢查訊息的接收者類,是 MyObject,然後分配一塊和我們的類同樣大小的記憶體空間,並初始化它的 isa 指標指向 MyObject 類,我們現在有了一個執行個體對象,最終把類對象的 +alloc 方法加入 NSObject 的類緩衝(class cache)中(lastly we put +alloc in NSObject's class cache for the class object )。

(3) 到現在為止,我們發送了一個類訊息,但是現在我們發送一個執行個體訊息,只是簡單的調用 -init 或者我們設計的初始化方法。當然,我們的類會響應這個方法,所以 -(id)init 加入到緩衝中。(譯註:要是 MyObject 實現了 init 方法,就會把 init 方法加入到 MyObject 的 class cache 中,要是沒有實現,只是因為繼承才有了這個方法,init 方法還是會加入到 NSObject 的 class cache 中)。

(4) 然後 self = [super init] 被調用。super 是個 magic keyword,指向對象的父類,所以我們得到了 NSObject 並調用它的的 init 方法。這樣可以確保 OOP(面相對象編程) 的繼承功能正常,這個方法可以正確的初始化父類的變數,之後你(在子類中)可以初始化自己的變數,如果需要可以覆蓋父類的方法。在 NSObject 的例子中,沒什麼重要的要做,但並不總是這樣。有時要做些重要的初始化。比如…


#import @interface MyObject : NSObject { NSString *aString; } @property(retain) NSString *aString; @end @implementation MyObject -(id)init { if (self = [super init]) { [self setAString:nil]; } return self; } @synthesize aString; @end int main (int argc, const char * argv[]) { NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init]; id obj1 = [NSMutableArray alloc]; id obj2 = [[NSMutableArray alloc] init]; id obj3 = [NSArray alloc]; id obj4 = [[NSArray alloc] initWithObjects:@"Hello",nil]; NSLog(@"obj1 class is %@",NSStringFromClass([obj1 class])); NSLog(@"obj2 class is %@",NSStringFromClass([obj2 class])); NSLog(@"obj3 class is %@",NSStringFromClass([obj3 class])); NSLog(@"obj4 class is %@",NSStringFromClass([obj4 class])); id obj5 = [MyObject alloc]; id obj6 = [[MyObject alloc] init]; NSLog(@"obj5 class is %@",NSStringFromClass([obj5 class])); NSLog(@"obj6 class is %@",NSStringFromClass([obj6 class])); [pool drain]; return 0; }
現在如果你新接觸 Cocoa ,我讓你猜會會輸出什麼,你可能會說:
NSMutableArrayNSMutableArray NSArrayNSArrayMyObjectMyObject
但實際上是:

obj1 class is __NSPlaceholderArray obj2 class is NSCFArray obj3 class is __NSPlaceholderArray obj4 class is NSCFArray obj5 class is MyObject obj6 class is MyObject


這是因為在 Objective-C 中 +alloc 方法可能會返回某個類的對象,然後在 -init 中返回另一個類的對象。
(譯註:感興趣的同學可以看下這兩篇文章:Class Clusters,Make Your Own Abstract Factory Class Cluster in Objective-C, 第二篇文章需要自備小梯子。)

七、那麼在 objc_msgSend 中發生了什嗎?
事實上在 objc_msgSend() 中發生了許多事兒。假設我們有這樣的代碼…


[self printMessageWithString:@"Hello World!"];
它實際上會被編譯器翻譯為…
objc_msgSend(self,@selector(printMessageWithString:),@"Hello World!");

我們沿著目標對象的 isa 指標尋找,看看是否這個對象響應 @selector(printMessageWithString:) selector。假設我們在類的分發表或者緩衝中找到了這個 selector,我們沿著函數指標並且執行它。這樣 objcmsgSend() 就永遠不會返回,它開始執行,然後沿著指向方法的指標,然後你的方法返回,這樣看起來 objcmsgSend() 方法返回了。Bill Bumgarner 比我講了更多 objc_msgSend() 的細節(部分1,部分2和部分3)。

概括下他說的,並且你已經看過了 Objective-C 的 runtime 代碼…

檢查忽略的 Selector 和短路(Short Circut)—— 顯然,如果我們運行在記憶體回收機制下,我們可以忽略調用 -retain, -release, 等等。

檢查 nil 對象(target)。和其他的語言不一樣的是,在 OC 中向 nil 發送訊息是完全合法的,並且有些原因下你會願意這麼做的。假設我們有個非 nil 的對象,然後我們繼續…

然後我們需要在這個類上找到 IMP,所以我們先從 class cache 中找起,如果找到了就沿著指標跳到這個函數。

如果沒有在緩衝中找到 IMP,然後去尋找類的分發表,如果找到了,就沿著指標跳到這個函數。

如果 IMP 沒有在緩衝和類的分發表中找到,然後我們跳到轉寄機制。這意味著最終你的代碼被編譯器轉換為 C 函數。你寫的方法會像這樣…

-(int)doComputeWithNum:(int)aNum
會被譯為:
int aClass_doComputeWithNum(aClass *self,SEL _cmd,int aNum)
OC Runtime 通過調用(invoking)指向這些方法的函數指標調用你的方法(call your methods)。現在,我要說的是,你不能直接調用這些被翻譯的方法,但是 Cocoa 架構提供了獲得函數指標的方法…
int (computeNum *)(id,SEL,int);//methodForSelector is COCOA & not ObjC Runtime//gets the same function pointer objc_msgSend getscomputeNum = (int (*)(id,SEL,int))[target methodForSelector:@selector(    doComputeWithNum:)]; //execute the C function pointer returned by the runtimecomputeNum(obj,@selector(doComputeWithNum:),aNum);
通過這種方法,你可以直接存取這個函數,並且可以在運行時直接調用,甚至可以使用這個避開 runtime 的動態特性,如果你絕對需要確保一個方法被執行。Objective-C 就是用這種途徑去調用你的方法的,但是使用的是 objc_msgSend()。

八、Objective-C 訊息轉寄
在 Objective-C 中向一個不知道如何響應這個方法的對象發送訊息是完全合法的(甚至可能是一種潛在的設計決定)。蘋果的文檔中給出的一個原因是類比多繼 承,Objective-C 不是原生支援的,或者你可能只是想抽象你的設計並且隱藏幕後處理這些訊息的其他對象/類。這一點是 runtime 非常需要的。它是這樣做的

1. Runtime 檢查了你的類和所有父類的 class cache 和分發表,但是沒找到指定的方法。

2. OC 的 Runtime 會在你的類上調用 + (BOOL) resolveInstanceMethod:(SEL)aSEL。 這就給了你一個機會去提供一個方法實現並且告訴 runtime 你已經解析了這個方法,如果它開始尋找,這回就會找到這個方法。你可以像這樣實現…定義一個函數…

void fooMethod(id obj, SEL _cmd){     NSLog(@"Doing Foo");}
然後你可以像這樣使用 class_addMethod() 解析它…
+(BOOL)resolveInstanceMethod:(SEL)aSEL{    if(aSEL == @selector(doFoo:))    {            class_addMethod([self class],aSEL,(IMP)fooMethod,"v@:");            return YES;    }    return [super resolveInstanceMethod];}
在 class_addMethod() 最後一部分的 "v@:" 是方法的返回和參數類型。你可以在 Runtime Guide 的 Type Encoding 章節看到完整介紹。

3. Runtime 然後調用 – (id)forwardingTargetForSelector:(SEL)aSelector。這樣做是為了給你一次機會(因為我們不能解析這個方法 (參見上面的 #2))引導 Objective-C runtime 到另一個可以響應這個訊息的對象上,在花費昂貴的處理程序呼叫 – (void)forwardInvocation:(NSInvocation *)anInvocation 之前調用這個方法也是更好的。你可以像這樣實現

- (id)forwardingTargetForSelector:(SEL)aSelector{    if(aSelector == @selector(mysteriousMethod:))    {                return alternateObject;    }    return [super forwardingTargetForSelector:aSelector];}
顯然你不想從這個方法直接返回 self,否則可能會產生一個死迴圈。

4. Runtime 最後一次會嘗試在目標對象上調用 – (void)forwardInvocation:(NSInvocation *)anInvocation。如果你從沒看過 NSInvocation,它是 Objective-C 訊息的對象形式。一旦你有了一個 NSInvocation 你可以改變這個訊息的一切,包括目標對象,selector 和參數。所以你可以這樣做…

-(void)forwardInvocation:(NSInvocation *)invocation{      SEL invSEL = invocation.selector;        if([altObject respondsToSelector:invSEL]) {                [invocation invokeWithTarget:altObject];        } else {                [self doesNotRecognizeSelector:invSEL];        }}
如果你繼承自 NSObject,預設它的 – (void)forwardInvocation:(NSInvocation *)anInvocation 實現只是簡單的調用 -doesNotRecognizeSelector:,你可以在最後一次機會裡覆蓋這個方法去做一些事情。(譯註:對這塊內容有興趣的同學可以參見:http://www.cnblogs.com/biosli/p/NSObjectinherit2.html)
九、Non Fragile ivars(Modern Runtime)(非脆弱的 ivar)

我們最近在 Modern Runtime 裡得到的是 Non Fragile ivars 的概念。當編譯你的類時,編譯器產生了一個 ivar 布局,顯示了在你的類中從哪可以訪問你的 ivars,擷取指向你的對象的指標,查看 ivar 與對象起始位元組的位移關係,和擷取讀入的變數類型的總共位元組大小等一些底層的細節。所以你的 ivar 布局可能看起來像這樣,左側的數字是位元組位移量。

我們有了 NSObject 的 ivar 布局,然後我們繼承自 NSObject 去擴充它並且添加了我們自己的 ivars。在蘋果發布更新前這都工作的很好,但是 Mac OS X 10.6 發布後,就成了這樣

 

你的自訂對象被剔除了因為我們有了一個重疊的父類。唯一可以防止這個的辦法是如果蘋果堅持之前的布局,如果他們這麼做了,那麼他們的架構就不能改進,因 為他們的 ivar 布局被凍住了。在 fragile ivar 下你不得不重新編譯你繼承自蘋果類的類來恢複相容性。所以在非 fragile ivar 時,會發生生嗎?

使用非 fragile ivars 時,編譯器產生和 fragile ivars 相同的 ivar 布局。然而當 runtime 檢測到一個重疊的超類時,它調整你在這個類中新增的 ivar 的位移量,這樣在子類中新增加的那部分就顯示出來了。

十、Objective-C 關聯對象

最近在 Mac OS X 10.6 雪豹 中新引入了關聯引用。Objective-C 不能動態添加一些屬性到對象上,和其他的一些原生支援這點的語言不一樣。所以之前你都不得不努力為未來要增加的變數預留好空間。在 Mac OS X 10.6 中,Objective-C 的 Runtime 已經原生的支援這個功能了。如果我們想向一個已有的類添加變數,看起來像這樣…


#import //Cocoa #include //objc runtime api’s @interface NSView (CustomAdditions) @property(retain) NSImage *customImage; @end @implementation NSView (CustomAdditions) static char img_key; //has a unique address (identifier) - (NSImage *)customImage { return objc_getAssociatedObject(self,&img_key); } - (void)setCustomImage:(NSImage *)image { objc_setAssociatedObject(self, &img_key,image, OBJC_ASSOCIATION_RETAIN); } @end objc_setAssociatedObject() 的選項,你可以在 runtime.h 檔案中找到。 /* Associated Object support. */ /* objc_setAssociatedObject() options */ enum { OBJC_ASSOCIATION_ASSIGN = 0, OBJC_ASSOCIATION_RETAIN_NONATOMIC = 1, OBJC_ASSOCIATION_COPY_NONATOMIC = 3, OBJC_ASSOCIATION_RETAIN = 01401, OBJC_ASSOCIATION_COPY = 01403 };

這些和 @property 文法中的選項意思一樣。

混和的 vTable Dispatch

如果你看過 modern runtime 的代碼,你會發現這個(在objc-runtime-new.m中)

/************************************************************************ vtable dispatch* * Every class gets a vtable pointer. The vtable is an array of IMPs.* The selectors represented in the vtable are the same for all classes*   (i.e. no class has a bigger or smaller vtable).* Each vtable index has an associated trampoline which dispatches to *   the IMP at that index for the receiver class's vtable (after *   checking for NULL). Dispatch fixup uses these trampolines instead *   of objc_msgSend.* Fragility: The vtable size and list of selectors is chosen at launch *   time. No compiler-generated code depends on any particular vtable *   configuration, or even the use of vtable dispatch at all.* Memory size: If a class's vtable is identical to its superclass's *   (i.e. the class overrides none of the vtable selectors), then *   the class points directly to its superclass's vtable. This means *   selectors to be included in the vtable should be chosen so they are *   (1) frequently called, but (2) not too frequently overridden. In *   particular, -dealloc is a bad choice.* Forwarding: If a class doesn't implement some vtable selector, that *   selector's IMP is set to objc_msgSend in that class's vtable.* +initialize: Each class keeps the default vtable (which always *   redirects to objc_msgSend) until its +initialize is completed.*   Otherwise, the first message to a class could be a vtable dispatch, *   and the vtable trampoline doesn't include +initialize checking.* Changes: Categories, addMethod, and setImplementation all force vtable *   reconstruction for the class and all of its subclasses, if the *   vtable selectors are affected.**********************************************************************/

背後的思想是,runtime 嘗試在這個 vtable 中儲存最近被調用的 selectors,這樣就可以提升你的應用的速度,因為它使用了比 objc_msgSend 更少的指令(fewer instructions)。vtable 中儲存 16 個全域最經常調用的 selectors,事實上順著代碼往下看你可以發現記憶體回收和非記憶體回收類型程式的預設 selectors :

 

static const char * const defaultVtable[] = {    "allocWithZone:",     "alloc",     "class",     "self",     "isKindOfClass:",     "respondsToSelector:",     "isFlipped",     "length",     "objectForKey:",     "count",     "objectAtIndex:",     "isEqualToString:",     "isEqual:",     "retain",     "release",     "autorelease", };static const char * const defaultVtableGC[] = {    "allocWithZone:",     "alloc",     "class",     "self",     "isKindOfClass:",     "respondsToSelector:",     "isFlipped",     "length",     "objectForKey:",     "count",     "objectAtIndex:",     "isEqualToString:",     "isEqual:",     "hash",     "addObject:",     "countByEnumeratingWithState:objects:count:", };
你可以在調試時從堆棧追蹤裡找到其中的method,可以像objc_msgSend()一樣將它們用於調試。
總結
OC Runtime是非常優秀的作品,它為支撐我們的Cocoa/Objective-C app以及眾多的優秀特性做了大量工作。你可以查看蘋果官方文檔來繼續深入瞭解(Objective-C Runtime Programming Guide、Objective-C Runtime Reference)。

最後說幾句,各位程式員們,上面的那些大多數部分是別人寫的,選擇原創是因為我找不到原創作者。。。。所以原創作者看到了的話不要說我盜用,給我留言,我會把出處標在最上方的。。。。還有真想學明白OC下的Runtime,我介意大家買一本叫《Effective Objective - C 2.0》的書。很不錯。。。。

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.