@dynamic属性及其用法?

3

有人能给我清晰地解释一下动态属性及其用法吗?为什么不能在所有地方都使用常规的@property呢?


可能是@synthesize与@dynamic有什么区别?的重复。 - Brad Larson
3个回答

6

动态属性是在编译时不提供实现,但确保在运行时存在实现的情况下使用的。作为一种动态语言,Objective-C可以在运行时响应消息,即使类在编译时没有实现。

这里举一个假设的例子:假设您有一个Book类,由一个包含键titleauthorNSMutableDictionary支持。但是,您希望Book也能够响应titleauthor并将它们作为属性;titleauthor将从字典中获取相应的值,而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 是动态属性。 (这不是最好的例子,但我认为它能够说明问题。)

这是对我来说最有意义的。 - kaala

1

@dynamic thing; 只是一种通知系统不要为 thing 生成 getter/setter 的方式,而是你(或其他人)将为你提供它们。也就是说,它们在运行时会存在。

这与 @synthesize 相反,后者告诉编译器为你生成 getter/setter(适当的话)。


2
请注意,@dynamic 是一个高级功能;当手动编写 getter/setter 的实现时,它并不是必需的。 - bbum

1

@dynamic(根据我的经验)主要与Core Data和NSManagedObject的子类一起使用。引用Marcus Zarra的Core Data中的话:

通过声明它们[属性/关系],我们告诉编译器忽略与这些属性相关的任何警告,因为我们“承诺”在运行时生成它们。当然,如果在运行时缺失了它们,那么我们的应用程序将会崩溃。


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接