用@dynamic或@synthesize实现@property有什么区别?


当前回答

@synthesize会为你的属性生成getter和setter方法。 @dynamic只是告诉编译器getter和setter方法不是由类本身实现的,而是在其他地方实现的(如超类或将在运行时提供)。

@dynamic的用途,例如NSManagedObject (CoreData)的子类,或者当你想为一个未被定义为outlet的超类定义的属性创建一个outlet时。

@dynamic也可以用来委派实现访问器的职责。如果在类中自己实现访问器,则通常不使用@dynamic。

超类:

@property (nonatomic, retain) NSButton *someButton;
...
@synthesize someButton;

子类:

@property (nonatomic, retain) IBOutlet NSButton *someButton;
...
@dynamic someButton;

其他回答

正如其他人所说,一般情况下,您使用@synthesize让编译器为您生成getter和/或设置,如果您要自己编写它们,则使用@dynamic。

还有一个尚未提及的微妙之处:@synthesize将允许您自己提供getter或setter的实现。如果您只想为一些额外的逻辑实现getter,而让编译器生成setter(对于对象来说,自己编写setter通常要复杂一些),那么这很有用。

然而,如果你确实为@synthesize'd访问器写了一个实现,它仍然必须有一个实字段作为支持(例如,如果你写-(int) getFoo();你必须有一个int foo;字段)。如果值是由其他东西产生的(例如,从其他字段计算),那么你必须使用@dynamic。

下面是@dynamic的例子

#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;
}

根据苹果文档。

在类的实现块中使用@synthesize语句告诉编译器创建与@property声明中给出的规范匹配的实现。

您可以使用@dynamic语句告诉编译器,如果它无法找到@property声明指定的访问器方法的实现,则取消警告。

更多信息:

https://developer.apple.com/library/ios/documentation/General/Conceptual/DevPedia-CocoaCore/DeclaredProperty.html

@dynamic通常在运行时动态创建属性时使用(如上所述)。NSManagedObject就是这样做的(为什么它的所有属性都是动态的)——这抑制了一些编译器警告。

有关如何动态创建属性(不含NSManagedObject和CoreData:)的良好概述,请参阅:http://developer.apple.com/library/ios/#documentation/Cocoa/Conceptual/ObjCRuntimeGuide/Articles/ocrtDynamicResolution.html#//apple_ref/doc/uid/TP40008048-CH102-SW1

@synthesize会为你的属性生成getter和setter方法。 @dynamic只是告诉编译器getter和setter方法不是由类本身实现的,而是在其他地方实现的(如超类或将在运行时提供)。

@dynamic的用途,例如NSManagedObject (CoreData)的子类,或者当你想为一个未被定义为outlet的超类定义的属性创建一个outlet时。

@dynamic也可以用来委派实现访问器的职责。如果在类中自己实现访问器,则通常不使用@dynamic。

超类:

@property (nonatomic, retain) NSButton *someButton;
...
@synthesize someButton;

子类:

@property (nonatomic, retain) IBOutlet NSButton *someButton;
...
@dynamic someButton;