@property 的本質(zhì)是什么?
@property = ivar + getter + setter;
下面解釋下:
“屬性” (property)有兩大概念:ivar(實(shí)例變量)清焕、存取方法(access method = getter + setter)并蝗。
“屬性” (property)作為 Objective-C 的一項(xiàng)特性,主要的作用就在于封裝對(duì)象中的數(shù)據(jù)秸妥。 Objective-C 對(duì)象通常會(huì)把其所需要的數(shù)據(jù)保存為各種實(shí)例變量滚停。實(shí)例變量一般通過(guò)“存取方法”(access method)來(lái)訪問(wèn)。其中粥惧,“獲取方法” (getter)用于讀取變量值键畴,而“設(shè)置方法” (setter)用于寫(xiě)入變量值。這個(gè)概念已經(jīng)定型突雪,并且經(jīng)由“屬性”這一特性而成為 Objective-C 2.0 的一部分起惕。 而在正規(guī)的 Objective-C 編碼風(fēng)格中,存取方法有著嚴(yán)格的命名規(guī)范咏删。 正因?yàn)橛辛诉@種嚴(yán)格的命名規(guī)范惹想,所以 Objective-C 這門(mén)語(yǔ)言才能根據(jù)名稱(chēng)自動(dòng)創(chuàng)建出存取方法。其實(shí)也可以把屬性當(dāng)做一種關(guān)鍵字督函,其表示:
編譯器會(huì)自動(dòng)寫(xiě)出一套存取方法嘀粱,用以訪問(wèn)給定類(lèi)型中具有給定名稱(chēng)的變量。 所以你也可以這么說(shuō):
@property = getter + setter;
ivar辰狡、getter锋叨、setter 是如何生成并添加到這個(gè)類(lèi)中的?
“自動(dòng)合成”( autosynthesis)
完成屬性定義后,編譯器會(huì)自動(dòng)編寫(xiě)訪問(wèn)這些屬性所需的方法搓译,此過(guò)程叫做“自動(dòng)合成”(autosynthesis)悲柱。需要強(qiáng)調(diào)的是,這個(gè)過(guò)程由編譯 器在編譯期執(zhí)行些己,所以編輯器里看不到這些“合成方法”(synthesized method)的源代碼豌鸡。除了生成方法代碼 getter、setter 之外段标,編譯器還要自動(dòng)向類(lèi)中添加適當(dāng)類(lèi)型的實(shí)例變量涯冠,并且在屬性名前面加下劃線,以此作為實(shí)例變量的名字逼庞。在前例中蛇更,會(huì)生成兩個(gè)實(shí)例變量,其名稱(chēng)分別為 _firstName 與 _lastName。也可以在類(lèi)的實(shí)現(xiàn)代碼里通過(guò) @synthesize 語(yǔ)法來(lái)指定實(shí)例變量的名字.
@implementation Person
@synthesize firstName = _myFirstName;
@synthesize lastName = _myLastName;
@end
我為了搞清屬性是怎么實(shí)現(xiàn)的,曾經(jīng)反編譯過(guò)相關(guān)的代碼,他大致生成了五個(gè)東西
OBJC_IVAR_$類(lèi)名$屬性名稱(chēng) :該屬性的“偏移量” (offset)派任,這個(gè)偏移量是“硬編碼” (hardcode)砸逊,表示該變量距離存放對(duì)象的內(nèi)存區(qū)域的起始地址有多遠(yuǎn)。
setter 與 getter 方法對(duì)應(yīng)的實(shí)現(xiàn)函數(shù)
ivar_list :成員變量列表
method_list :方法列表
prop_list :屬性列表
也就是說(shuō)我們每次在增加一個(gè)屬性,系統(tǒng)都會(huì)在 ivar_list 中添加一個(gè)成員變量的描述,在 method_list 中增加 setter 與 getter 方法的描述,在屬性列表中增加一個(gè)屬性的描述,然后計(jì)算該屬性在對(duì)象中的偏移量,然后給出 setter 與 getter 方法對(duì)應(yīng)的實(shí)現(xiàn),在 setter 方法中從偏移量的位置開(kāi)始賦值,在 getter 方法中從偏移量開(kāi)始取值,為了能夠讀取正確字節(jié)數(shù),系統(tǒng)對(duì)象偏移量的指針類(lèi)型進(jìn)行了類(lèi)型強(qiáng)轉(zhuǎn).