568

@property实现with@dynamic或之间有什么区别@synthesize

4

8 回答 8

749

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

@dynamic 的使用例如与NSManagedObject(CoreData) 的子类一起使用,或者当您想要为未定义为出口的超类定义的属性创建出口时。

@dynamic 也可用于委派实现访问器的责任。如果您自己在类中实现访问器,那么您通常不使用@dynamic。

超级班:

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

子类:

@property (nonatomic, retain) IBOutlet NSButton *someButton;
...
@dynamic someButton;
于 2009-07-21T17:21:33.833 回答
213

看看这篇文章;在“运行时提供的方法”标题下:

一些访问器是在运行时动态创建的,例如 CoreData 的 NSManagedObject 类中使用的某些访问器。如果您想为这些情况声明和使用属性,但又想避免在编译时出现有关方法缺失的警告,您可以使用@dynamic 指令而不是@synthesize。

...

使用@dynamic 指令本质上是告诉编译器“别担心,有一个方法在路上”。

@synthesize另一方面,该指令在编译时为您生成访问器方法(尽管如“混合合成访问器和自定义访问器”部分所述,它很灵活,并且不会为您生成方法,如果其中任何一个被实现)。

于 2009-07-21T17:20:06.300 回答
32

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

还有一个尚未提及的微妙之处:@synthesize让您自己提供一个实现,无论是 getter 还是 setter。如果您只想为一些额外的逻辑实现 getter,但让编译器生成 setter(对于对象来说,它通常自己编写起来有点复杂),这很有用。

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

于 2009-07-21T18:04:50.347 回答
14

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

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

于 2011-11-25T21:30:09.127 回答
14

这是@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;
}
于 2013-09-11T09:27:12.373 回答
11

根据文档:

https://developer.apple.com/library/mac/documentation/cocoa/conceptual/ObjCRuntimeGuide/Articles/ocrtDynamicResolution.html

@dynamic 告诉编译器访问器方法是在运行时提供的。

经过一番调查,我发现提供访问器方法会覆盖 @dynamic 指令。

@synthesize 告诉编译器为你创建这些访问器(getter 和 setter)

@property 告诉编译器将创建访问器,并且可以使用点符号或 [object message] 访问

于 2013-12-03T18:25:15.807 回答
6

要补充的一件事是,如果一个属性被声明为@dynamic,它将不会占用内存(我通过分配工具确认)。结果是您可以在类类别中声明属性。

于 2013-05-13T09:14:48.713 回答
4

根据 Apple 文档。

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

@dynamic如果它找不到由@property声明指定的访问器方法的实现,您可以使用该语句告诉编译器抑制警告。

更多信息:-

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

于 2015-12-16T11:21:55.227 回答