Відповіді:
@synthesize створить для вашої власності методи отримання та налаштування. @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. У мене не було відповідного ivar для мого @property). Додавання @dynamic
виправленої проблеми - збирається та працює нормально. Дякую!
@property
елементи, які не мають ні, @synthesize
ні @dynamic
будуть автоматично синтезовані. Для кожної властивості буде створено ivar з провідним підкресленням, наприклад _propertyName
, разом із відповідним геттером та сеттером.
Погляньте на цю статтю ; під заголовком "Методи, що надаються під час виконання":
Деякі аксесуари створюються динамічно під час виконання, наприклад, такі, які використовуються в класі NSManagedObject CoreData. Якщо ви хочете оголосити та використовувати властивості для цих випадків, але хочете уникати попереджень про відсутність методів під час компіляції, можете використовувати директиву @dynamic замість @synthesize.
...
Використання директиви @dynamic по суті каже компілятору "не хвилюйтеся з цього приводу, метод на шляху".
@synthesize
Директива, з іншого боку, створює методи доступу для вас під час компіляції (хоча , як відзначався в розділі «Змішування синтезатор і призначене для користувача Accessors» він є гнучким і не створює методи для вас , якщо або реалізується).
Як вже говорили інші, ти, як правило, використовуєш @synthesize, щоб компілятор генерував для тебе геть та / або налаштування, і @dynamic, якщо ти збираєшся їх самостійно писати.
Існує ще одна тонкість ще не згадав: @synthesize буде перешкоджати вам забезпечити реалізацію себе, або геттер або сетера. Це корисно, якщо ви хочете реалізувати геттер лише за деякою додатковою логікою, але дозвольте компілятору генерувати сеттер (що для об'єктів, як правило, трохи складніше написати самому).
Однак якщо ви пишете реалізацію для аксесуара @ synthesize'd, вона все одно повинна бути підкріплена реальним полем (наприклад, якщо ви пишете, у -(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 повідомляє компілятору створити для вас ці аксесуари (геттер і сетер)
@property повідомляє компілятору, що будуть створені аксесуари, і що їх можна отримати за допомогою позначення крапки або [об'єктного повідомлення]
Хочемо додати, що якщо властивість оголошено як @dynamic, воно не буде займати пам'ять (я підтвердив інструмент розподілу). Наслідком цього є те, що ви можете оголосити властивість у категорії класів.
Відповідно до документації Apple.
Ви використовуєте @synthesize
оператор у блоці реалізації класу, щоб повідомити компілятору створити реалізації, які відповідають специфікації, яку ви дали у@property
декларації.
Ви використовуєте @dynamic
оператор, щоб сказати компілятору придушити попередження, якщо він не може знайти реалізацію методів аксесуара, визначених @property
декларацією.
Більше інформації:-