2012-08-07 32 views
5

在Objective-C的舊版本,一個objc_class結構實現這樣的:實例變量和方法如何存儲在Objective-C 2.0對象中?

struct objc_class { 
    Class isa; 
    Class super_class; 
    const char *name; 
    long version; 
    long info; 
    long instance_size; 
    struct objc_ivar_list *ivars; 
    struct objc_method_list **methodLists; 
    struct objc_cache *cache; 
    struct objc_protocol_list *protocols; 
}; 

所以,表示對象的結構存儲一個指向對象的類,一個指向對象的超類,對象的類名,對象的版本,信息和實例大小,對象的實例變量列表,對象的方法列表,對象的緩存和對象的協議列表。表示對象的結構體中的這些字段的存在是非常容易理解的,因爲它們中的每一個存儲關於該對象的信息。

然而,Objective-C的2.0實現相同的結構objc_class的是這樣的:

struct objc_class { 
    Class isa; 
}; 

所以,在這個版本objc_class的,只有一個在結構領域:一個指向對象的類結構。

那麼我的問題是,如何存儲在Objective-C 2.0中的對象的其他信息,因爲結構中只有一個字段表示對象?

+3

有趣閱讀:[\ [objc解釋\]:非脆弱的ivars](http://www.sealiesoftware.com/blog/archive/2009/01/27/objc_explain_Non-fragile_ivars.html) – 2012-08-07 00:18:25

回答

7

這一切都在新的(好吧,以及不那麼新了)非脆弱的ABI。

基本上,不是像過去那樣將iVar存儲在一個結構體內(如果超類改變它的iVar佈局,將會中斷繼承),編譯器會將iVar重定向到另一個層,類似於運行時的objc_setAssociatedObject

這允許一些有趣的場景。考慮以下幾點:

@interface A { // part of libA.a 
    id var1; 
    int var2; 
    float var3; 
} 

@end 

@interface B : A { // part of libB.a 
    id var4; 
} 

@end 

現在,如果一段時間在路上,我們需要改變A類,我們確定我們需要在var3更精確(將其轉換爲long double,例如)?

在舊的,脆弱的ABI中,我們將被擰緊,直到libB的製造商更新。但是,通過這個新的非易碎ABI,我們可以靈活地改變這一切,而libB仍然有效。

儘管在理論上這可能會慢幾個週期,但它增加了在運行時查找iVars的更簡單方法,更靈活的子類化以及支持不同種類的iVars(例如__weak)。

+0

確定。因此,在Objective-C的「新」版本中,而不是在編譯時填充結構體,對象ivars和methodos是通過像'class_addIvar'和'class_addMethod'這樣的函數在運行時存儲的? – LuisABOL 2012-08-07 01:35:30

+0

@laboleite所有的意圖和目的,是的,他們是。實際上,它們仍然作爲結構存儲,但它們的偏移量是變化的。 – 2012-08-07 01:38:38

+0

但是,這些結構屬於運行時系統,並由上述功能填充,對嗎? – LuisABOL 2012-08-07 01:49:39