Динамические свойства используются, когда вы не предоставляете реализацию во время компиляции, но гарантируете, что она существует во время выполнения.Будучи динамическим языком, Objective-C может отвечать на сообщения во время выполнения, даже если у класса нет реализации во время компиляции.
Вот надуманный пример: допустим, у вас есть класс Book
,поддержанный NSMutableDictionary
, который содержит ключи title
и author
.Однако вы хотите, чтобы Book
также отвечал на title
и author
и имел их в качестве свойств;title
и author
извлекут соответствующее значение из словаря, а setTitle:
и setAuthor:
изменит значение, сохраненное в словаре.Вы можете сделать это с помощью этого кода :
#import <Foundation/Foundation.h>
@interface Book : NSObject
{
NSMutableDictionary *data;
}
@property (retain) NSString *title;
@property (retain) NSString *author;
@end
@implementation Book
@dynamic title, author;
- (id)init
{
if ((self = [super init])) {
data = [[NSMutableDictionary alloc] init];
[data setObject:@"Tom Sawyer" forKey:@"title"];
[data setObject:@"Mark Twain" forKey:@"author"];
}
return self;
}
- (void)dealloc
{
[data release];
[super dealloc];
}
- (NSMethodSignature *)methodSignatureForSelector:(SEL)selector
{
NSString *sel = NSStringFromSelector(selector);
if ([sel rangeOfString:@"set"].location == 0) {
return [NSMethodSignature signatureWithObjCTypes:"v@:@"];
} else {
return [NSMethodSignature signatureWithObjCTypes:"@@:"];
}
}
- (void)forwardInvocation:(NSInvocation *)invocation
{
NSString *key = NSStringFromSelector([invocation selector]);
if ([key rangeOfString:@"set"].location == 0) {
key = [[key substringWithRange:NSMakeRange(3, [key length]-4)] lowercaseString];
NSString *obj;
[invocation getArgument:&obj atIndex:2];
[data setObject:obj forKey:key];
} else {
NSString *obj = [data objectForKey:key];
[invocation setReturnValue:&obj];
}
}
@end
int main(int argc, char **argv)
{
NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
Book *book = [[Book alloc] init];
printf("%s is written by %s\n", [book.title UTF8String], [book.author UTF8String]);
book.title = @"1984";
book.author = @"George Orwell";
printf("%s is written by %s\n", [book.title UTF8String], [book.author UTF8String]);
[book release];
[pool release];
return 0;
}
Обратите внимание, что методы "создаются" во время выполнения с помощью forwardInvocation:
;следовательно, title
и author
- это динамические свойства.
(Это не лучший пример, но я думаю, что это правильно.)