Отправлять и получать сообщения через NSNotificationCenter в Objective-C?


610

Я пытаюсь отправлять и получать сообщения через NSNotificationCenterObjective-C. Однако я не смог найти примеров того, как это сделать. Как вы отправляете и получаете сообщения через NSNotificationCenter?


Действительно очень полезно, спасибо. Во-первых, метод addObserver не должен иметь точку с запятой после указанного селектора (по крайней мере, это вызвало исключение в моей версии этого). Я попытался отредактировать код выше, но изменение не было принято из-за проблем с форматированием в исходном коде.
Брауний,

3
Это было здорово: cocoawithlove.com/2008/06/…
Арам Кочарян

2
этот вопрос слишком простой и широкий, немного погулять было бы хорошо
Daij-Djan

Это очень похоже на связанный с этим вопрос здесь: stackoverflow.com/questions/7896646/…
Дэвид Дуглас

55
Я нахожу абсурдным такой вопрос, как закрытый и неконструктивный, когда пользователи Stack Overflow так четко прокомментировали его полезность
Чет

Ответы:


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?
фульвио

14
@Fulvio: зависит, если вы получаете или публикуете уведомления, которые потенциально могут повлиять на все части вашего приложения, поместите его в свой AppDelegate. Если вы получаете / публикуете уведомления, которые влияют только на один класс, поместите его в этот класс.
Dreamlax

1
@dreamlax Правда, однако стоит заметить, потому что этот вопрос в основном ищут новые разработчики ios, которые поддерживают прослушиватель уведомлений дольше, чем им нужно. Теперь с дугой вы обычно не используете dealloc, и в результате некоторые могут подумать, что им не нужно освобождать слушателя.
Вив

7
Также стоит упомянуть, что [super dealloc]вызов в методе dealloc не разрешен в ARC; остальное все хорошо.
Томмис

1
Что произойдет, если уведомление сработало, а наблюдателей нет? Уведомление потеряно? Или он «сохранен» где-то готовым к отправке новому наблюдателю (созданному позже)?
суперпуччо

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
Могу ли я получить доступ к наблюдателю selfв методе receiveTestNotification?
почему

почему да. receiveTestNotification - это метод экземпляра, и у вас есть доступ к самому экземпляру через себя внутри него.
Майкл Петерсон

Вот и все. Я искал способ получить UserInfo из метода получателя.
Хасан

Кажется, что вся эта идея наблюдателя не охватывает все случаи. это не сработало, когда приложение. был закрыт, и форма уведомления центр уведомлений была нажата. метод наблюдателя не вызывается.
Хасан

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


Это сработало для меня! Спасибо
Ракшита Муранга Родриго

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

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
поток, из которого вы отправляете уведомление, является потоком, выполняющим функции и, таким образом, пытающимся изменить пользовательский интерфейс. вы также можете использовать диспетчеризацию для основного потока внутри функций, как вы сказали: D. должен иметь тот же результат, кажется, он даже лучше: D
eiran

1
@eiran, спасибо большое, братан, это сработало только после того, как я написал в dispatch_async
Аршад

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)
 }
Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.