Objective-CのNSNotificationCenterを介してメッセージを送受信しますか?


610

NSNotificationCenterObjective-Cでメッセージを送受信しようとしています。しかし、私はこれを行う方法の例を見つけることができませんでした。どのようにしてメッセージを送受信しますNSNotificationCenterか?


本当にとても感謝しています。1つは、addObserverメソッドでは、指定されたセレクターの後にセミコロンを付けないようにする必要があります(少なくとも、このバージョンでは例外が発生していました)。上記のコードを編集してみましたが、元のコードのフォーマットの問題により、変更は受け入れられませんでした。
Braunius

3
これは素晴らしかった:cocoawithlove.com/2008/06/...
アラムKocharyan

2
このqはあまりにも基本的で広範であり、少しグーグル化するのは良いことだと思います
Daij-Djan

これは、ここに関連する質問に非常に似ていますstackoverflow.com/questions/7896646/...
デビッド・ダグラス

55
私は、スタックオーバーフローのユーザーはそれほど明確にその有用コメントしているとき、それは、このような不条理Aの質問は建設的ではない閉じられた見つける
チェット

回答:


1019
@implementation TestClass

- (void) dealloc
{
    // If you don't remove yourself as an observer, the Notification Center
    // will continue to try and send notification objects to the deallocated
    // object.
    [[NSNotificationCenter defaultCenter] removeObserver:self];
    [super dealloc];
}

- (id) init
{
    self = [super init];
    if (!self) return nil;

    // Add this instance of TestClass as an observer of the TestNotification.
    // We tell the notification center to inform us of "TestNotification"
    // notifications using the receiveTestNotification: selector. By
    // specifying object:nil, we tell the notification center that we are not
    // interested in who posted the notification. If you provided an actual
    // object rather than nil, the notification center will only notify you
    // when the notification was posted by that particular object.

    [[NSNotificationCenter defaultCenter] addObserver:self
        selector:@selector(receiveTestNotification:) 
        name:@"TestNotification"
        object:nil];

    return self;
}

- (void) receiveTestNotification:(NSNotification *) notification
{
    // [notification name] should always be @"TestNotification"
    // unless you use this method for observation of other notifications
    // as well.

    if ([[notification name] isEqualToString:@"TestNotification"])
        NSLog (@"Successfully received the test notification!");
}

@end

...別のクラスのどこかで...

- (void) someMethod
{

    // All instances of TestClass will be notified
    [[NSNotificationCenter defaultCenter] 
        postNotificationName:@"TestNotification" 
        object:self];

}

2
[NSNotificationCenter defaultCenter]がどこに配置されるのかを考えているだけです。AppDelegateに配置するのが最善ですか?
fulvio 2011年

14
@Fulvio:アプリケーションのすべての部分に影響を与える可能性のある通知を受信または投稿している場合は、それをAppDelegateに入れてください。単一のクラスのみに影響する通知を受信/投稿する場合は、代わりにそのクラスに配置してください。
dreamlax

1
@dreamlax Truth。ただし、この質問は、通知リスナーを必要以上に長く存続させる新しいiOS開発者が主に検索するため、注目に値します。現在、arcでは通常deallocを使用しないため、リスナーを解放する必要がないと考える人もいます。
2013年

7
また[super dealloc]、dealloc-methodの呼び出しはARCでは許可されていないことにも言及する価値があります。残りはすべて良いです。
トミー

1
通知が発生し、オブザーバーがいない場合はどうなりますか?通知は失われますか?または、新しいオブザーバー(後で作成)に出荷する準備ができている場所に「保存」されていますか?
superpuccio

226

dreamlaxの例を拡張する ...通知と一緒にデータを送信する場合

投稿コード:

NSDictionary *userInfo = 
[NSDictionary dictionaryWithObject:myObject forKey:@"someKey"];
[[NSNotificationCenter defaultCenter] postNotificationName: 
                       @"TestNotification" object:nil userInfo:userInfo];

コードを観察する場合:

- (void) receiveTestNotification:(NSNotification *) notification {

    NSDictionary *userInfo = notification.userInfo;
    MyObject *myObject = [userInfo objectForKey:@"someKey"];
}

TestNotificationはNSStringタイプである必要があります。それはインスタンス変数NSNotificationですか?
RomanHouse

1
selfreceiveTestNotificationメソッドでオブザーバーにアクセスできますか?
なぜ

なぜいいの。receiveTestNotificationはインスタンスメソッドであり、インスタンス内のselfを介してインスタンス自体にアクセスできます。
マイケルピーターソン

それでおしまい。レシーバーメソッドからUserInfoを取得する方法を探していました。
ハサン、2015年

そのオブザーバーの考えはすべてのケースをカバーしているわけではないようです。これはアプリのときに機能しませんでした。が閉じられ、通知フォームから通知センターがタップされました。オブザーバーメソッドは呼び出されません。
ハサン、2015年

49

これは私を助けました:

// Add an observer that will respond to loginComplete
[[NSNotificationCenter defaultCenter] addObserver:self 
                                             selector:@selector(showMainMenu:) 
                                                 name:@"loginComplete" object:nil];


// Post a notification to loginComplete
[[NSNotificationCenter defaultCenter] postNotificationName:@"loginComplete" object:nil];


// the function specified in the same class where we defined the addObserver
- (void)showMainMenu:(NSNotification *)note {
    NSLog(@"Received Notification - Someone seems to have logged in"); 
}

出典:http : //www.smipple.net/snippet/Sounden/Simple%20NSNotificationCenter%20example


それは私のために働いた!ありがとう
Rakshitha Muranga Rodrigo

48

ブロックを使用する可能性もあります:

NSOperationQueue *mainQueue = [NSOperationQueue mainQueue];
[[NSNotificationCenter defaultCenter] 
     addObserverForName:@"notificationName" 
     object:nil
     queue:mainQueue
     usingBlock:^(NSNotification *notification)
     {
          NSLog(@"Notification received!");
          NSDictionary *userInfo = notification.userInfo;

          // ...
     }];

Appleのドキュメント


1
これは私の回答の良いアップデートで、かなり古いものです。導入またはARCとブロックを使用すると、通知センターの処理がはるかに簡単になります。
dreamlax 2013

5
私もそう考えましたが、本当であるにはあまりにも良いことがわかりました。この場合、addObserverが返すオブザーバーを保持し、後でそれを削除する必要があるため、新しいメソッドを作成するのと同じくらい複雑になります。さらに詳しい情報:toastmo.com/blog/2012/12/04/...
アンドリュー

42

ビューの更新にNSNotificationCenterを使用している場合は、メインスレッドから次のように呼び出して、ビューを送信することを忘れないでくださいdispatch_async

dispatch_async(dispatch_get_main_queue(),^{
    [[NSNotificationCenter defaultCenter] postNotificationName:@"my_notification" object:nil];
});

1
メインスレッドから発生する必要があるのは通知ポストですか、それとも実際にビューを更新したとき、つまりメインスレッドにディスパッチする通知を受信するメソッド内ですか?
Crashalot

1
通知を送信するスレッドは、関数を実行しているスレッドなので、UIを変更しようとします。あなたが言ったように、関数内のメインスレッドへのディスパッチを使用することもできます:D。同じ結果になるはずです、それはさらに良いと思います:D
eiran '19 / 04/16

1
@eiranさん、本当にありがとうございました。dispatch_asyncの中に書き込んだ後でのみ機能しました
Arshad

2

初心者向けの選択された回答のSWIFT 5.1

class TestClass {
    deinit {
        // If you don't remove yourself as an observer, the Notification Center
        // will continue to try and send notification objects to the deallocated
        // object.
        NotificationCenter.default.removeObserver(self)
    }

    init() {
        super.init()

        // Add this instance of TestClass as an observer of the TestNotification.
        // We tell the notification center to inform us of "TestNotification"
        // notifications using the receiveTestNotification: selector. By
        // specifying object:nil, we tell the notification center that we are not
        // interested in who posted the notification. If you provided an actual
        // object rather than nil, the notification center will only notify you
        // when the notification was posted by that particular object.

        NotificationCenter.default.addObserver(self, selector: #selector(receiveTest(_:)), name: NSNotification.Name("TestNotification"), object: nil)
    }

    @objc func receiveTest(_ notification: Notification?) {
        // [notification name] should always be @"TestNotification"
        // unless you use this method for observation of other notifications
        // as well.

        if notification?.name.isEqual(toString: "TestNotification") != nil {
            print("Successfully received the test notification!")
        }
    }
}

...別のクラスのどこかで...

 func someMethod(){
        // All instances of TestClass will be notified
        NotificationCenter.default.post(name: "TestNotification", object: self)
 }
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.