關(guān)聯(lián)是指把兩個(gè)對(duì)象相互關(guān)聯(lián)起來,使得其中的一個(gè)對(duì)象作為另外一個(gè)對(duì)象的一部分昵仅。
- 關(guān)聯(lián)特性只有在Mac OS X V10.6以及以后的版本上才是可用的退渗。
- 使用關(guān)聯(lián),我們可以不用修改類的定義而為其對(duì)象增加存儲(chǔ)空間犀勒。這在我們無法訪問到類的源碼的時(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)致資源不可回收)褂萧。
- 使用關(guān)聯(lián)時(shí)會(huì)涉及到3個(gè)OC的運(yùn)行時(shí)(runtime)函數(shù):objc_setAssociatedObject押桃、objc_getAssociatedObject、objc_removeAssociatedObjects
創(chuàng)建關(guān)聯(lián)
- 創(chuàng)建關(guān)聯(lián)要使用objc_setAssociatedObject函數(shù)來把一個(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)策略。
- 關(guān)鍵字是一個(gè)void類型的指針磕昼。每一個(gè)關(guān)聯(lián)的關(guān)鍵字必須是唯一的壳坪。通常都是會(huì)采用靜態(tài)變量來作為關(guān)鍵字。
- 關(guān)聯(lián)策略表明了相關(guān)的對(duì)象是通過賦值掰烟、保留引用還是復(fù)制的方式進(jìn)行關(guān)聯(lián)的爽蝴;還有這種關(guān)聯(lián)是原子的還是非原子的。這里的關(guān)聯(lián)策略和聲明屬性時(shí)的很類似纫骑。這種關(guān)聯(lián)策略是通過使用預(yù)先定義好的常量來表示的蝎亚。
下面展示如何把一個(gè)字符串關(guān)聯(lián)到一個(gè)數(shù)組上。
static char overviewKey;
NSArray *array = [[NSArray alloc] initWithObjects:@"One", @"Two", @"Three", nil];
// 為了演示先馆,這里使用initWithFormat:來確保字符串可以被銷毀
NSString *overview = [[NSString alloc] initWithFormat:@"numbers"];
objc_setAssociatedObject(array, &overviewKey, overview, OBJC_ASSOCIATION_RETAIN);
[overview release];
// ①此時(shí)overview仍然可用
NSLog(@"%@", overview);
[array release];
// ②此時(shí)overview不可用
- 在①處发框,字符串overview仍然是可用的,這是因?yàn)镺BJC_ASSOCIATION_RETAIN策略指明了數(shù)組要保有相關(guān)的對(duì)象煤墙。關(guān)聯(lián)后梅惯,字符串的引用計(jì)數(shù)為2。
- 在②處仿野,當(dāng)數(shù)組array被銷毀的時(shí)候铣减,overview也就會(huì)被釋放,其引用計(jì)數(shù)為0脚作,因此被銷毀葫哗。如果此時(shí)還想使用overview,例如輸出overview的值球涛,則會(huì)出現(xiàn)運(yùn)行時(shí)異常劣针。
獲取被關(guān)聯(lián)的對(duì)象
獲取被關(guān)聯(lián)的對(duì)象時(shí)使用objc_getAssociatedObject函數(shù):
NSString *associatedObject = objc_getAssociatedObject(array, &overviewKey);
NSLog(@"associatedObject = %@", associatedObject);
斷開關(guān)聯(lián)
斷開關(guān)聯(lián)是使用objc_setAssociatedObject函數(shù),傳入nil值即可:
objc_setAssociatedObject(array, &overviewKey, nil, OBJC_ASSOCIATION_ASSIGN);
- 其中亿扁,被關(guān)聯(lián)的對(duì)象為nil捺典,此時(shí)關(guān)聯(lián)策略也就無關(guān)緊要了。
- 使用函數(shù)objc_removeAssociatedObjects可以斷開所有關(guān)聯(lián)从祝。通常情況下不建議使用這個(gè)函數(shù)襟己,因?yàn)樗鼤?huì)斷開所有關(guān)聯(lián)。只有在需要把對(duì)象恢復(fù)到“原始狀態(tài)”的時(shí)候才會(huì)使用這個(gè)函數(shù)哄褒。
下面是完整demo:
#import <Foundation/Foundation.h>
#import <objc/runtime.h>
int main(int argc, const char * argv[])
{
static char overviewKey;
NSArray *array = [[NSArray alloc] initWithObjects:@"One", @"Two", @"Three", nil];
// 為了演示稀蟋,這里使用initWithFormat:來確保字符串可以被銷毀
NSString *overview = [[NSString alloc] initWithFormat:@"numbers"];
objc_setAssociatedObject(array, &overviewKey, overview, OBJC_ASSOCIATION_RETAIN);
[overview release];
// 此時(shí)overview仍然可用
NSLog(@"%@", overview);
NSString *associatedObject = objc_getAssociatedObject(array, &overviewKey);
NSLog(@"associatedObject = %@", associatedObject);
objc_setAssociatedObject(array, &overviewKey, nil, OBJC_ASSOCIATION_ASSIGN);
NSString *newObject = objc_getAssociatedObject(array, &overviewKey);
// 輸出為空
NSLog(@"newObject = %@", newObject);
return 0;
}
關(guān)聯(lián)