@property
实现with@dynamic
或之间有什么区别@synthesize
?
8 回答
@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;
看看这篇文章;在“运行时提供的方法”标题下:
一些访问器是在运行时动态创建的,例如 CoreData 的 NSManagedObject 类中使用的某些访问器。如果您想为这些情况声明和使用属性,但又想避免在编译时出现有关方法缺失的警告,您可以使用@dynamic 指令而不是@synthesize。
...
使用@dynamic 指令本质上是告诉编译器“别担心,有一个方法在路上”。
@synthesize
另一方面,该指令在编译时为您生成访问器方法(尽管如“混合合成访问器和自定义访问器”部分所述,它很灵活,并且不会为您生成方法,如果其中任何一个被实现)。
正如其他人所说,通常您使用@synthesize 让编译器为您生成getter 和/或设置,如果您要自己编写它们,则使用@dynamic。
还有一个尚未提及的微妙之处:@synthesize将让您自己提供一个实现,无论是 getter 还是 setter。如果您只想为一些额外的逻辑实现 getter,但让编译器生成 setter(对于对象来说,它通常自己编写起来有点复杂),这很有用。
但是,如果您确实为@synthesize'd 访问器编写了一个实现,它仍然必须由一个真实的字段支持(例如,如果您编写-(int) getFoo();
您必须有一个int foo;
字段)。如果该值是由其他东西产生的(例如从其他字段计算),那么您必须使用@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 告诉编译器将创建访问器,并且可以使用点符号或 [object message] 访问
要补充的一件事是,如果一个属性被声明为@dynamic,它将不会占用内存(我通过分配工具确认)。结果是您可以在类类别中声明属性。
根据 Apple 文档。
您使用类的实现块中的语句来告诉编译器创建与您在声明@synthesize
中给出的规范相匹配的实现。@property
@dynamic
如果它找不到由@property
声明指定的访问器方法的实现,您可以使用该语句告诉编译器抑制警告。
更多信息:-