objective-c 2.0中增加了一個新的關鍵字@dynamic, 用於定義動態屬性。所謂動態屬性相對於@synthesis,不是由編譯器自動產生setter或者getter,也不是由開發人員自己寫的setter或getter,而是在運行時動態添加的setter和getter。
一般我們定義一個屬性都是類似以下方法:
@interface Car:NSObject;@property (retain) NSString* name;@end@implement Car;@synthesize name;@end
這種情況下,@synthesize關鍵字告訴編譯器自動實現setter和getter。另外,如果不使用@synthesize,也可以自己實現getter或者setter
@implement Car;(NSString*) name{return _name;}(void) setName:(NSString*) n{_name = n;}
現在通過@dynamic,還可以通過第三種方法來實現name的setter和getter。實現動態屬性,需要在代碼中覆蓋resolveInstanceMethod來動態添加name的setter和getter。這個方法在每次找不到方法實現的時候都會被調用。事實上,NSObject的預設實現就是拋出異常。
參考以下代碼:
下面是定義動態屬性和實現動態屬性的代碼:
@interface Car:NSObject@property (retain) NSString* name;@end---car.m(void) dynamicSetName(id SELF, SEL _cmd, NSString * n){//這個定義形式是必須的。//結合下面的類型描述字元v表示返回為void//@表示第一個參數id//:表示SEL//@表示參數nNSLog(@"the new name is going to send in:%@!", n);}@implement Car@dynamic name;-(BOOL) resolveInstanceMethod:(SEL) sel{NSString * method = NSStringFromSelector(sel);if([method isEqualToString:@"setName:"]){class_addMethod([self class], sel, (IMP)dynamicSetName, "v@:@");//類型描述字元,可以參考開發文檔中有關@encode關鍵字的說明。return YES:}return [super resolveInstanceMethod:sel];}@end;
上面的代碼動態實現了name的setter,當然這個時候如果想要調用car.name,就會拋出錯誤,因為我並沒有實現它的getter。