關聯
關聯是指把兩個對象互相關聯起來,使得其中的一個對象作為另外一個對象的一部分。
關聯特性隻有在Mac OS X V10.6以及以後的版本上才是可用的。
使用關聯,我們可以不用修改類的定義而為其對象增加存儲空間。這在我們無法通路到類的源碼的時候或者是考慮到二進制相容性的時候是非常有用。
關聯是基于關鍵字的,是以,我們可以為任何對象增加任意多的關聯,每個都使用不同的關鍵字即可。關聯是可以保證被關聯的對象在關聯對象的整個生命周期都是可用的(在垃圾自動回收環境下也不會導緻資源不可回收)。
建立關聯要使用到Objective-C的運作時函數:objc_setAssociatedObject來把一個對象與另外一個對象進行關聯。該函數需要四個參數:源對象,關鍵字,關聯的對象和一個關聯政策。當然,此處的關鍵字和關聯政策是需要進一步讨論的。
■ 關鍵字是一個void類型的指針。每一個關聯的關鍵字必須是唯一的。通常都是會采用靜态變量來作為關鍵字。
■ 關聯政策表明了相關的對象是通過指派,保留引用還是複制的方式進行關聯的;還有這種關聯是原子的還是非原子的。這裡的關聯政策和聲明屬性時的很類似。這種關聯政策是通過使用預先定義好的常量來表示的。
下面的代碼展示了如何把一個字元串關聯到一個數組上。
清單7-1 把一個字元串關聯到一個數組
[cpp] view plaincopy
static char overviewKey;
NSArray * array =[[NSArray alloc] initWidthObjects:@"One", @"Two", @"Three", nil];
//為了示範的目的,這裡使用initWithFormat:來確定字元串可以被銷毀
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仍然是可用的,這是因為OBJC_ASSOCIATION_RETAIN政策指明了數組要保有相關的對象。當數組array被銷毀的時候,也就是在(2)處overview也就會被釋放,是以而被銷毀。如果此時還想使用overview,例如想通過log來輸出overview的值,則會出現運作時異常。
擷取相關聯的對象時使用Objective-C函數objc_getAssociatedObject。接着上面清單7-1的代碼,我們可以使用如下代碼來擷取與array相關聯的字元串:
NSString * associatedObject = (NSString *)objc_getAssociatedObject(array, &oveviewKey);
斷開關聯是使用objc_setAssociatedObject函數,傳入nil值即可。
接着清單7-1中的程式,我們可以使用如下的代碼來斷開字元串overview和arry之間的關聯:
objc_setAssociatedObject(array, &overviewKey, nil, OBJC_ASSOCIATION_ASSIGN);
其中,被關聯的對象為nil,此時關聯政策也就無關緊要了。
使用函數objc_removeAssociatedObjects可以斷開所有關聯。通常情況下不建議使用這個函數,因為他會斷開所有關聯。隻有在需要把對象恢複到“原始狀态”的時候才會使用這個函數。
下面的程式綜合了前面的代碼.
#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:來確定字元串可以被銷毀
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;
}