想到要如何為所有的對象增加執行個體變量嗎?我們知道,使用Category可以很友善地為現有的類增加方法,但卻無法直接增加執行個體變量。不過從Mac OS X v10.6開始,系統提供了Associative References,這個問題就很容易解決了。這種方法也就是所謂的關聯(association),我們可以在runtime期間動态地添加任意多的屬性,并且随時讀取。所用到的兩個重要runtime API是:
OBJC_EXPORT void objc_setAssociatedObject(id object, const void *key, id value, objc_AssociationPolicy policy)
OBJC_EXPORT id objc_getAssociatedObject(id object, const void *key)
__OSX_AVAILABLE_STARTING(__MAC_10_6, __IPHONE_3_1);
現在我們結合一個實際的例子來說明他們的用法。假設我們現在打算利用category對UILabel進行屬性補充,添加FlashColor屬性。一般我們有個原則:能用category擴充就不用繼承,因為随着繼承深度的增加,代碼的可維護性也會增加很多。使用category可以這麼做:
#import <UIKit/UIKit.h>
#import <objc/runtime.h>
@interface UILabel (Associate)//單單從頭檔案看是不是很像一個類?再看看.m檔案你就知道這些都是假象了
- (nonatomic, retain) UIColor *FlashColor;
@end
#import "UILabel+Associate.h"
@implementation UILabel (Associate)
@dynamic FlashColor;
static char flashColorKey;
- (void) setFlashColor:(UIColor *) flashColor{
objc_setAssociatedObject(self, &flashColorKey, flashColor, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
- (UIColor *) getFlashColor{
return objc_getAssociatedObject(self, &flashColorKey);
}
@end
上面的例子有幾個需要注意的地方:
1、key:我們注意到在函數簽名中key的類型const void *,這表示key僅僅是一個位址,而不是字元串的内容,這也是為說明flashColorKey沒有初始化的原因,因為具體指向什麼内容我們無所謂,我們要的僅僅是位址!如果在setAssocaitedObject中你傳入的是flashColorKey,那get方法得到的值将會是nil。正确的應該是傳入位址&flashColorKey。
2、policy:這裡的policy跟屬性聲明中的retain、assign、copy是一樣的,不再贅述
3、在implement開始處的@dynamic聲明。一般來說@dynamic與@synthesize都可以用來聲明屬性,@synthesize是預設的聲明,意思是編譯器在編譯階段自動為你的屬性生成setter與getter;而@dynamic則告訴編譯器,别慌,小子,編譯階段不用為我生成setter與getter,在runtime我已經自己實作了setter與getter。此處我們選擇@dynamic。事實上,二者曾引起stackOverFlow上強烈的争論:請點這裡。
下面我們再來看另一個例子,來源于APPLE GUIDE:
#import <Foundation/Foundation.h>
#import <objc/runtime.h>
int main (int argc, const char * argv[]) {
@autoreleasepool {
/*Seciton 0. 關聯資料的Key和Value*/
static char overviewKey;
static const char *myOwnKey = "VideoProperty\0";
static const char intValueKey = 'i';
NSArray *array = [[NSArray alloc]
initWithObjects:@ "One", @"Two", @"Three", nil];
// For the purposes of illustration, use initWithFormat: to ensure
// we get a deallocatable string
NSString *overview = [[NSString alloc]
initWithFormat:@"%@", @"First three numbers"];
NSString *videoKeyValue = @"This is a video";
NSNumber *intValue = [[NSNumber alloc]initWithInt:5];
/*Section 1. 關聯資料設定部分*/
objc_setAssociatedObject (
array,
&overviewKey,
overview,
OBJC_ASSOCIATION_RETAIN
);
[overview release];
objc_setAssociatedObject (
array,
myOwnKey,
videoKeyValue,
OBJC_ASSOCIATION_RETAIN
);
objc_setAssociatedObject (
array,
&intValueKey,
intValue,
OBJC_ASSOCIATION_RETAIN
);
/*Section 3. 關聯資料查詢部分*/
NSString *associatedObject = (NSString *) objc_getAssociatedObject (array, &overviewKey);
NSLog(@"associatedObject: %@", associatedObject);
NSString *associatedObject2 = (NSString *) objc_getAssociatedObject(array, myOwnKey);
NSLog(@"Video Key value is %@", associatedObject2);
NSString *assObject3 = (NSString *) objc_getAssociatedObject(array, &myOwnKey);
if( assObject3 )
{
NSLog(@"不會進入這裡! assObject3 應當為nil!");
}
else
{
NSLog(@"OK. 通過myOwnKey的位址是得不到資料的!");
}
NSNumber *assKeyValue = (NSNumber *) objc_getAssociatedObject(array, &intValueKey);
NSLog(@"Int value is %d",[assKeyValue intValue]);
/*Section 3. 關聯資料清理部分*/
objc_setAssociatedObject (
array,
&overviewKey,
nil,
OBJC_ASSOCIATION_ASSIGN
);
objc_setAssociatedObject (
array,
myOwnKey,
nil,
OBJC_ASSOCIATION_ASSIGN
);
objc_setAssociatedObject (
array,
&intValueKey,
nil,
OBJC_ASSOCIATION_ASSIGN
);
[array release];
}
return 0;
}
=======================================================
原創文章,轉載請注明 程式設計小翁@部落格園,郵件[email protected],歡迎各位與我在C/C++/Objective-C/機器視覺等領域展開交流!
=======================================================