關(guān)聯(lián)
- 關(guān)聯(lián)是指把兩個(gè)對(duì)象相互關(guān)聯(lián)起來(lái),使得其中的一個(gè)對(duì)象作為另外一個(gè)對(duì)象的一部分。
- 關(guān)聯(lián)特性只有在Mac OS X V10.6以及以后的版本上才是可用的尼变。
在類(lèi)的定義之外為類(lèi)增加額外的存儲(chǔ)空間
使用關(guān)聯(lián),我們可以不用修改類(lèi)的定義而為其對(duì)象增加存儲(chǔ)空間。這在我們無(wú)法訪問(wèn)到類(lèi)的源碼的時(shí)候或者是考慮到二進(jìn)制兼容性的時(shí)候是非常有用乡洼。
關(guān)聯(lián)是基于關(guān)鍵字的,因此匕坯,我們可以為任何對(duì)象增加任意多的關(guān)聯(lián)束昵,每個(gè)都使用不同的關(guān)鍵字即可。關(guān)聯(lián)是可以保證被關(guān)聯(lián)的對(duì)象在關(guān)聯(lián)對(duì)象的整個(gè)生命周期都是可用的(在垃圾自動(dòng)回收環(huán)境下也不會(huì)導(dǎo)致資源不可回收)葛峻。
創(chuàng)建關(guān)聯(lián)
創(chuàng)建關(guān)聯(lián)要使用到Objective-C的運(yùn)行時(shí)函數(shù):objc_setAssociatedObject來(lái)把一個(gè)對(duì)象與另外一個(gè)對(duì)象進(jìn)行關(guān)聯(lián)锹雏。該函數(shù)需要四個(gè)參數(shù):源對(duì)象,關(guān)鍵字术奖,關(guān)聯(lián)的對(duì)象和一個(gè)關(guān)聯(lián)策略礁遵。當(dāng)然,此處的關(guān)鍵字和關(guān)聯(lián)策略是需要進(jìn)一步討論的采记。
- 關(guān)鍵字是一個(gè)void類(lèi)型的指針佣耐。每一個(gè)關(guān)聯(lián)的關(guān)鍵字必須是唯一的。通常都是會(huì)采用靜態(tài)變量來(lái)作為關(guān)鍵字唧龄。
- 關(guān)聯(lián)策略表明了相關(guān)的對(duì)象是通過(guò)賦值兼砖,保留引用還是復(fù)制的方式進(jìn)行關(guān)聯(lián)的;還有這種關(guān)聯(lián)是原子的還是非原子的既棺。這里的關(guān)聯(lián)策略和聲明屬性時(shí)的很類(lèi)似讽挟。這種關(guān)聯(lián)策略是通過(guò)使用預(yù)先定義好的常量來(lái)表示的。
下面的代碼展示了如何把一個(gè)字符串關(guān)聯(lián)到一個(gè)數(shù)組上丸冕。
列表7-1 把一個(gè)字符串關(guān)聯(lián)到一個(gè)數(shù)組
static char overviewKey;
NSArray * array =[[NSArray alloc] initWidthObjects:@"One", @"Two", @"Three", nil];
//為了演示的目的耽梅,這里使用initWithFormat:來(lái)確保字符串可以被銷(xiāo)毀
NSString * overview = [[NSString alloc] initWithFormat:@"@",@"First three numbers"];
objc_setAssociatedObject(array, &overviewKey, overview, OBJC_ASSOCIATION_RETAIN);
[overview release];
//(1) overview仍然是可用的
[array release];
//(2)overview 不可用
在(1)處,字符串overview仍然是可用的胖烛,這是因?yàn)镺BJC_ASSOCIATION_RETAIN策略指明了數(shù)組要保有相關(guān)的對(duì)象眼姐。當(dāng)數(shù)組array被銷(xiāo)毀的時(shí)候,也就是在(2)處overview也就會(huì)被釋放洪己,因此而被銷(xiāo)毀妥凳。如果此時(shí)還想使用overview,例如想通過(guò)log來(lái)輸出overview的值答捕,則會(huì)出現(xiàn)運(yùn)行時(shí)異常逝钥。
獲取相關(guān)聯(lián)的對(duì)象
獲取相關(guān)聯(lián)的對(duì)象時(shí)使用Objective-C函數(shù)objc_getAssociatedObject。接著上面列表7-1的代碼,我們可以使用如下代碼來(lái)獲取與array相關(guān)聯(lián)的字符串:
NSString * associatedObject = (NSString *)objc_getAssociatedObject(array, &oveviewKey);
斷開(kāi)關(guān)聯(lián)
斷開(kāi)關(guān)聯(lián)是使用objc_setAssociatedObject函數(shù)艘款,傳入nil值即可持际。
接著列表7-1中的程序,我們可以使用如下的代碼來(lái)斷開(kāi)字符串overview和arry之間的關(guān)聯(lián):
objc_setAssociatedObject(array, &overviewKey, nil, OBJC_ASSOCIATION_ASSIGN);
其中哗咆,被關(guān)聯(lián)的對(duì)象為nil蜘欲,此時(shí)關(guān)聯(lián)策略也就無(wú)關(guān)緊要了。
使用函數(shù)objc_removeAssociatedObjects可以斷開(kāi)所有關(guān)聯(lián)晌柬。通常情況下不建議使用這個(gè)函數(shù)姥份,因?yàn)樗麜?huì)斷開(kāi)所有關(guān)聯(lián)。只有在需要把對(duì)象恢復(fù)到“原始狀態(tài)”的時(shí)候才會(huì)使用這個(gè)函數(shù)年碘。
一個(gè)完整的實(shí)例程序
下面的程序綜合了前面的代碼.
#import <Foundation/Foundation.h>
#import <objc/runtime.h>
int main(int argc, const char* argv[])
{
NSAutoreleasePool * pool = [[NSAutoreleasePool] alloc init];
static char overviewKey;
NSArray *array =[[NSArray alloc] initWidthObjects:@"One", @"Two", @"Three", nil];
//為了演示的目的澈歉,這里使用initWithFormat:來(lái)確保字符串可以被銷(xiāo)毀
NSString * overview = [[NSString alloc] initWithFormat:@"@",@"First three numbers"];
objc_setAssociatedObject(array, &overviewKey, overview, OBJC_ASSOCIATION_RETAIN);
[overview release];
NSString *associatedObject = (NSString *)objc_getAssociatedObject(arrray, &overviewKey);
NSLog(@"associatedObject:%@", associatedObject);
objc_setAssociatedObject(array, &overviewKey, nil, OBJC_ASSOCIATION_ASSIGN);
[array release];
[pool drain];
return 0;
}
tips: 使用objc_setAssociatedObject,需要引入頭文件 #import <objc/runtime.h>