Wysyłać i odbierać wiadomości za pośrednictwem NSNotificationCenter w Objective-C?

610

Próbuję wysyłać i odbierać wiadomości NSNotificationCenterw Objective-C. Nie znalazłem jednak żadnych przykładów, jak to zrobić. Jak wysyłasz i odbierasz wiadomości NSNotificationCenter?

hichris123
źródło
Naprawdę bardzo przydatne, dzięki. Jedna rzecz: metoda addObserver nie powinna mieć końcowego średnika po określonym selektorze (przynajmniej spowodowało wyjątek w mojej wersji tego). Próbowałem edytować powyższy kod, ale zmiana nie została zaakceptowana z powodu problemów z formatowaniem oryginalnego kodu.
Braunius
3
To było świetne: cocoawithlove.com/2008/06/…
Aram
2
to q jest zbyt proste i szerokie, trochę googleingu byłoby lepiej
Daij-Djan
Jest to bardzo podobne do pokrewnego pytania tutaj: stackoverflow.com/questions/7896646/…
David Douglas
55
Uważam, że to absurdalne pytanie, jak to jest zamknięte a nie konstruktywny gdy użytkownicy przepełnienie stosu nie tak wyraźnie skomentował swoją przydatność
Chet

Odpowiedzi:

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

... gdzieś indziej w innej klasie ...

- (void) someMethod
{

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

}
dreamlax
źródło
2
Zastanawiam się tylko, gdzie należy umieścić [NSNotificationCenter defaultCenter]. Czy najlepiej umieścić go w AppDelegate?
fulvio 12.01.11
14
@Fulvio: Zależy, jeśli otrzymujesz lub publikujesz powiadomienia, które potencjalnie wpływają na wszystkie części aplikacji, umieść je w AppDelegate. Jeśli otrzymujesz / publikujesz powiadomienia, które dotyczą tylko jednej klasy, umieść ją w tej klasie.
dreamlax
1
@dreamlax Truth, jednak warto zauważyć, ponieważ to pytanie jest przeszukiwane głównie przez nowych deweloperów ios, którzy utrzymują odbiorcę powiadomień dłużej niż potrzebują. Teraz z łukiem zazwyczaj nie używasz dealloc, w wyniku czego niektórzy mogą myśleć, że nie muszą zwalniać słuchacza.
Przeżyj
7
Warto również wspomnieć, że [super dealloc]wywołanie metody dealloc nie jest dozwolone na podstawie ARC; cała reszta jest dobra.
tommys
1
Co się stanie, jeśli powiadomienie zostanie uruchomione i nie będzie żadnych obserwatorów? Zgubiono powiadomienie? A może jest „zapisany” gdzieś gotowy do wysłania do nowego obserwatora (utworzonego później)?
superpuccio
226

Aby rozwinąć na przykładzie dreamlax ... Jeśli chcesz wysłać dane wraz z powiadomieniem

W kodzie pocztowym:

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

W przestrzeganiu kodu:

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

    NSDictionary *userInfo = notification.userInfo;
    MyObject *myObject = [userInfo objectForKey:@"someKey"];
}
Michael Peterson
źródło
TestNotification musi być typu NSString. Czy jest to zmienna instancji NSNotification?
RomanHouse
1
Czy mogę uzyskać dostęp do obserwatora selfza pomocą metody receiveTestNotification?
dlaczego
dlaczego - tak receiveTestNotification to metoda instancji, a Ty masz dostęp do samej instancji poprzez siebie w niej.
Michael Peterson
Otóż ​​to. Szukałem sposobu na uzyskanie UserInfo z metody odbiornika.
hasan
Wygląda na to, że cały ten pomysł obserwatora nie obejmuje wszystkich przypadków. to nie działało, gdy aplikacja. został zamknięty i powiadomienie z centrum powiadomień zostało stuknięte. metoda obserwatora nie zostaje wywołana.
hasan
49

Ten pomógł mi:

// 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"); 
}

Źródło: http://www.smipple.net/snippet/Sounden/Simple%20NSNotificationCenter%20example

j7nn7k
źródło
To działało dla mnie! Dzięki
Rakshitha Muranga Rodrigo
48

Istnieje również możliwość użycia bloków:

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

          // ...
     }];

Dokumentacja Apple

Xavi Gil
źródło
1
To dobra aktualizacja mojej odpowiedzi, która jest dość przestarzała. Dzięki wprowadzeniu lub ARC i blokom centra powiadomień są znacznie łatwiejsze w obsłudze.
dreamlax
5
Też tak myślałem, ale okazuje się, że to zbyt piękne, aby mogło być prawdziwe. W takim przypadku musisz zatrzymać obserwatora, który zwraca addObserver, a później go usunąć, co czyni to tak skomplikowanym, jak tworzenie nowej metody, jeśli nie bardziej. Więcej informacji: toastmo.com/blog/2012/12/04/…
Andrew
42

jeśli używasz NSNotificationCenter do aktualizacji widoku, nie zapomnij wysłać go z głównego wątku, wywołując dispatch_async:

dispatch_async(dispatch_get_main_queue(),^{
    [[NSNotificationCenter defaultCenter] postNotificationName:@"my_notification" object:nil];
});
Eiran
źródło
1
czy jest to powiadomienie, które musi pojawić się w głównym wątku, czy tylko po zaktualizowaniu widoku, tj. w metodzie otrzymywania powiadomienia wysyłanego do głównego wątku?
Crashalot 19.04.16
1
wątek, z którego wysyłane jest powiadomienie, to wątek uruchamiający funkcje i próbujący w ten sposób zmienić interfejs użytkownika. możesz także użyć wysyłki do głównego wątku wewnątrz funkcji, tak jak powiedziałeś: D. powinien mieć ten sam wynik, perheps, jest jeszcze lepiej: D
Eiran 19.04.16
1
@eiran, dziękuję bardzo bracie, zadziałało to dopiero po napisaniu w środku dispatch_async
Arshad Shaik
2

SWIFT 5.1 wybranej odpowiedzi dla początkujących

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!")
        }
    }
}

... gdzieś indziej w innej klasie ...

 func someMethod(){
        // All instances of TestClass will be notified
        NotificationCenter.default.post(name: "TestNotification", object: self)
 }
Maneesh M.
źródło