[Objective-C]關聯(objc_setAssociatedObject、objc_getAssociatedObject、objc_removeAssociatedObjects),associatedobjects
關聯
關聯是指把兩個對象相互關聯起來,使得其中的一個對象作為另外一個對象的一部分。
關聯特性只有在Mac OS X V10.6以及以後的版本上才是可用的。
在類的定義之外為類增加額外的儲存空間
使用關聯,我們可以不用修改類的定義而為其對象增加儲存空間。這在我們無法訪問到類的源碼的時候或者是考慮到二進位相容性的時候是非常有用。
關聯是基於關鍵字的,因此,我們可以為任何對象增加任意多的關聯,每個都使用不同的關鍵字即可。關聯是可以保證被關聯的對象在關聯對象的整個生命週期都是可用的(在垃圾自動回收環境下也不會導致資源不可回收)。
建立關聯
建立關聯要使用到Objective-C的運行時函數:objc_setAssociatedObject來把一個對象與另外一個對象進行關聯。該函數需要四個參數:來源物件,關鍵字,關聯的對象和一個關聯策略。當然,此處的關鍵字和關聯策略是需要進一步討論的。
■ 關鍵字是一個void類型的指標。每一個關聯的關鍵字必須是唯一的。通常都是會採用靜態變數來作為關鍵字。
■ 關聯策略表明了相關的對象是通過賦值,保留引用還是複製的方式進行關聯的;還有這種關聯是原子的還是非原子的。這裡的關聯策略和聲明屬性時的很類似。這種關聯策略是通過使用預先定義好的常量來表示的。
下面的代碼展示了如何把一個字串關聯到一個數組上。
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; }