@synthesizeと@dynamicの違いは何ですか?


回答:


744

@synthesizeは、プロパティのゲッターメソッドとセッターメソッドを生成します。@dynamicは、getterメソッドとsetterメソッドがクラス自体ではなく他の場所(スーパークラスのように、または実行時に提供される)によって実装されることをコンパイラーに伝えます。

@dynamicの使用は、たとえばNSManagedObject(CoreData)のサブクラスを使用する場合、またはアウトレットとして定義されていないスーパークラスによって定義されたプロパティのアウトレットを作成する場合です。

@dynamicを使用して、アクセサの実装の責任を委任することもできます。クラス内で自分でアクセサーを実装する場合、通常は@dynamicを使用しません。

スーパークラス:

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

サブクラス:

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

25
100%正しいわけではありません。@synthesizeまたは@dynamicのいずれも設定しない場合のデフォルトはdynamicです。@dynamicを指定することは、プロパティ宣言のシグネチャに基づいてプロパティアクセサーを適切に実装する責任を負うことを意味します。
ケブラー

68
実際にはそうではありませんが、@動的な手段は、アクセサを実装する責任を委任されています。クラス内で自分でアクセサーを実装する場合、通常は@dynamicを使用しません。
diederikh 2009

2
NSUnknownKeyExceptionを削除すると、動的プロパティでエラーが発生しました@synthesize(Xcode 3.2でエラーb / cが発生しましたが、@ propertyに対応するivarがありませんでした)。@dynamic修正された問題を追加すると、コンパイルして正常に実行できます。ありがとう!
pix0r 2009

4
申し訳ありませんが、これを購入することは完全に間違っています。@dynamicは、クラスまたはスーパークラス(他の場所ではない)で宣言されていない限り、アクセサーは実行時に解決されることを示します。ドキュメントdeveloper.apple.com/library/mac/documentation/cocoa/conceptual/…
user1447414

5
ケブラー:いいえ。最近のObjCでは、自動合成さ@propertyれていない、@synthesizeまたは@dynamic自動合成されないアイテム。各プロパティについて、アンダースコアが先行するivar _propertyNameが、適切なゲッターとセッターと共に作成されます。
デイブR

212

見てみましょうこの記事を。「実行時に提供されるメソッド」という見出しの下:

CoreDataのNSManagedObjectクラスで使用される特定のアクセサなど、一部のアクセサは実行時に動的に作成されます。これらの場合にプロパティを宣言して使用したいが、コンパイル時にメソッドが見つからないという警告を避けたい場合は、@ synthesizeの代わりに@dynamicディレクティブを使用できます。

...

@dynamicディレクティブを使用すると、本質的にコンパイラーに「心配する必要はありません。メソッドが作成中です」と伝えます。

@synthesizeディレクティブは、他の一方で、(「合成され、カスタムアクセサをミキシング」の項で述べたようものの、それは柔軟性があり、実装されているいずれかの場合にはあなたのためのメソッドを生成しません)コンパイル時にあなたのためのアクセサメソッドを生成します。


27
これはより正確な人です。この回答は、実行時に作成されるメソッドについて話す唯一の回答です。これは、実際にトップ投票のans atmよりもはるかに多くの精神を捉えているようです
bobobobo 2009年

30

他の人が言ったように、一般に@synthesizeを使用してコンパイラにゲッターや設定を生成させ、それらを自分で作成する場合は@dynamicを使用します。

まだ言及されていない別の微妙があります:@synthesizeはなりますが、ゲッターやセッターのいずれかで、実装を自分で提供してみましょう。これは、いくつかの追加ロジックのゲッターを実装するだけで、コンパイラーにセッターを生成させる場合に役立ちます(オブジェクトの場合、通常、自分で書くのは少し複雑です)。

ただし、@ synthesize'dアクセサーの実装を作成する場合は、実際のフィールドを使用-(int) getFoo();する必要があります(たとえば、作成する場合はint foo;フィールドが必要です)。値が他の何かによって生成されている場合(他のフィールドから計算されている場合など)、@ dynamicを使用する必要があります。


2
重要な違いについて言及する場合は+1:@dynamicを使用すると、クラスインターフェースで定義されていない変数のアクセサを作成し、イントロスペクションを実行できます。
mahboudz 2009年

24
「そして@dynamic、あなたがそれらを自分で書くつもりなら」いいえ、あなたがそれらを自分で書く場合、動的を使用しません。@dynamicコンパイラーのチェックをオフにして、それらが実装されていることを確認します。それらを自分で実装した場合は、コンパイラーにチェックさせてください。
user102008、2011

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


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

10

ドキュメントに従って:

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

@dynamicは、アクセサメソッドが実行時に提供されることをコンパイラに通知します。

少し調べたところ、アクセサメソッドを提供すると@dynamicディレクティブがオーバーライドされることがわかりました。

@synthesizeは、これらのアクセサーを作成するようコンパイラーに指示します(getterおよびsetter)。

@propertyは、アクセサーが作成され、ドット表記または[オブジェクトメッセージ]でアクセスできることをコンパイラーに通知します


6

追加したいことの1つは、プロパティが@dynamicとして宣言されている場合、メモリを占有しないことです(割り当てツールで確認しました)。結果として、クラスカテゴリでプロパティを宣言できます。


カテゴリ内のプロパティセッターをオーバーライドして動的にした場合、これにより、親クラスのセッターではなく、オーバーライドが実行時に使用されることが保証されますか?Apple docsから:「カテゴリで宣言されたメソッドの名前が元のクラスのメソッドと同じである場合...実行時にどのメソッド実装が使用されるかについての動作は未定義です。」
デビッドジェームズ

いいえ、動作はまだ定義されていないと思います。カテゴリのプロパティを動的にしても、プロパティセッターメソッドの実行時の優先順位は変更されません。
Yingpei Zeng 2015

3

Appleのドキュメントに従って。

@synthesizeクラスの実装ブロックでステートメントを使用して、@property宣言で指定した仕様に一致する実装を作成するようコンパイラーに指示します。

@dynamicステートメントを使用して、@property宣言で指定されたアクセサーメソッドの実装が見つからない場合に警告を抑制するようコンパイラーに指示します。

より詳しい情報:-

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

弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.