Answers:
@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;
NSUnknownKeyException
的错误与我的动态特性,当我删除了@synthesize
行(Xcode的3.2是给我一个错误B / C我为我的@property没有匹配伊娃)。添加@dynamic
解决了该问题-现在可以编译并正常运行。谢谢!
@property
既没有@synthesize
也@dynamic
不会自动合成的项目。对于每个属性,_propertyName
将创建带有下划线的ivar,例如适当的getter和setter。
看一下这篇文章 ; 在“运行时提供的方法”标题下:
一些访问器是在运行时动态创建的,例如CoreData的NSManagedObject类中使用的某些访问器。如果要声明和使用这些情况下的属性,但又希望避免在编译时对方法丢失的警告,则可以使用@dynamic指令而不是@synthesize。
...
使用@dynamic指令实际上告诉编译器“不用担心,一种方法即将出现。”
@synthesize
另一方面,该指令在编译时为您生成访问器方法(尽管如“混合的综合和自定义访问器”一节中所述,它很灵活,如果实现了两者,则不会为您生成方法)。
正如其他人所说,通常,您使用@synthesize来让编译器为您生成getter和/或设置,如果要自己编写,则使用@dynamic。
还没有提到另一个微妙之处:@synthesize 将允许您自己提供getter或setter的实现。如果您只想为某些额外的逻辑实现getter,但让编译器生成setter(对于对象,编写自己通常要复杂一点),这将很有用。
但是,如果您确实为@synthesize访问器编写了一个实现,则该实现仍必须由实数字段支持(例如,如果编写,则-(int) getFoo();
必须具有一个int foo;
字段)。如果该值是由其他值产生的(例如,从其他字段计算得出),则必须使用@dynamic。
@dynamic
如果您要自己编写”,如果您自己编写,则不要使用动态。@dynamic
关闭编译器检查以确保您实现了它们。如果您自己实现它们,则确实希望编译器进行检查。
当在运行时动态创建属性时,通常使用@dynamic(如上所述)。NSManagedObject这样做(为什么其所有属性都是动态的)-抑制了一些编译器警告。
有关如何动态创建属性(没有NSManagedObject和CoreData :)的良好概述,请参阅:http : //developer.apple.com/library/ios/#documentation/Cocoa/Conceptual/ObjCRuntimeGuide/Articles/ocrtDynamicResolution.html#// apple_ref / doc / uid / TP40008048-CH102-SW1
这是 @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;
}
根据文档:
@dynamic告诉编译器在运行时提供了访问器方法。
经过一点调查,我发现提供访问器方法会覆盖@dynamic指令。
@synthesize告诉编译器为您创建那些访问器(getter和setter)
@property告诉编译器将创建访问器,并且可以使用点符号或[对象消息]进行访问。
根据Apple文档。
您可以@synthesize
在类的实现块中使用该语句来告诉编译器创建与您在@property
声明中给出的规范匹配的实现。
您可以使用该@dynamic
语句告诉编译器如果找不到@property
声明指定的访问器方法的实现,则禁止显示警告。
更多信息:-