@ динамические свойства и их использование?

Кто-нибудь может дать мне ясную картину о динамическом свойстве и его использовании? вы не используете обычный @property везде?

3 ответа

Решение

Динамические свойства используются, когда вы не предоставляете реализацию во время компиляции, но гарантируете, что она существует во время выполнения. Будучи динамическим языком, 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 являются динамическими свойствами.

(Это не лучший пример, но я думаю, что это понятно.)

@dynamic thing; это просто способ информировать систему не генерировать геттеры / сеттеры для thingчто вы (или кто-то еще) предоставят их вам - как, они будут там во время выполнения.

Это в отличие от @synthesize который говорит компилятору сгенерировать getter/setter (при необходимости) для вас.

@dynamic (по моему опыту) используется в основном в сочетании с Core Data и подклассами NSManagedObject. Цитируя основные данные Маркуса Зарры,

Объявляя их [атрибуты / отношения], мы говорим компилятору игнорировать любые предупреждения, связанные с их свойствами, потому что мы "обещаем" генерировать их во время выполнения. Естественно, если во время выполнения происходит сбой, то наше приложение будет зависать.

Другие вопросы по тегам