Inviare e ricevere messaggi tramite NSNotificationCenter in Objective-C?


610

Sto tentando di inviare e ricevere messaggi attraverso NSNotificationCenterin Objective-C. Tuttavia, non sono stato in grado di trovare esempi su come farlo. Come si inviano e si ricevono i messaggi NSNotificationCenter?


Davvero molto utile, grazie. Una cosa, il metodo addObserver non dovrebbe avere il punto e virgola finale dopo il selettore specificato (almeno ha causato un'eccezione nella mia versione di questo). Ho provato a modificare il codice sopra ma la modifica non è stata accettata a causa di problemi di formattazione nel codice originale.
Braunius,

3
Questo è stato grandioso: cocoawithlove.com/2008/06/…
Aram Kocharyan,

2
questo q è troppo semplice e ampio, un po 'di google sarebbe stato bello
Daij-Djan

Questo è molto simile a una domanda correlata qui: stackoverflow.com/questions/7896646/…
David Douglas,

55
Trovo assurdo che una domanda come questa sia chiusa e non costruttiva quando gli utenti di Stack Overflow hanno commentato così chiaramente la sua utilità
Chet

Risposte:


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

... altrove in un'altra classe ...

- (void) someMethod
{

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

}

2
Mi chiedo solo dove [NSNotificationCenter defaultCenter] dovrebbe essere posizionato. È meglio inserirlo nel tuo AppDelegate?
Fulvio,

14
@Fulvio: Dipende, se stai ricevendo o pubblicando notifiche che potenzialmente influenzano tutte le parti della tua applicazione, inseriscilo nel tuo AppDelegate. Se ricevi / pubblichi notifiche che riguardano solo una singola classe, inseriscile invece in quella classe.
dreamlax,

1
@dreamlax La verità, tuttavia, vale la pena notare perché questa domanda è per lo più cercata dai nuovi sviluppatori iOS che mantengono in vita l'ascoltatore delle notifiche più a lungo del necessario. Ora con arc di solito non usi dealloc e di conseguenza alcuni potrebbero pensare di non dover rilasciare l'ascoltatore.
Vive

7
Potrebbe anche valere la pena ricordare che la [super dealloc]chiamata nel metodo dealloc non è consentita in ARC; il resto va bene.
Tommy,

1
Cosa succede se la notifica si attiva e non ci sono osservatori? La notifica è andata persa? O è "salvato" da qualche parte pronto per essere spedito a un nuovo osservatore (creato in seguito)?
superpuccio,

226

Per espandere l'esempio di dreamlax ... Se si desidera inviare dati insieme alla notifica

Nel codice di invio:

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

Nell'osservare il codice:

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

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

TestNotification deve essere di tipo NSString. È una variabile di istanza NSNotification?
RomanHouse,

1
Posso accedere all'osservatore selfcon il metodo receiveTestNotification?
perché il

perchè si. receiveTestNotification è un metodo di istanza e hai accesso all'istanza stessa tramite sé al suo interno.
Michael Peterson,

Questo è tutto. Stavo cercando un modo per ottenere UserInfo dal metodo del ricevitore.
Hasan,

Sembra che tutta quell'idea di osservatore non copra tutti i casi. questo non ha funzionato quando l'app. è stato chiuso e un modulo di notifica è stato toccato il centro notifiche. il metodo osservatore non viene chiamato.
Hasan,

49

Questo mi ha aiutato:

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

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


Ha funzionato per me! Grazie
Rakshitha Muranga Rodrigo il

48

C'è anche la possibilità di usare i blocchi:

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

          // ...
     }];

Documentazione di Apple


1
Questo è un buon aggiornamento della mia risposta che ora è abbastanza obsoleto. Con l'introduzione o ARC e blocchi, i centri di notifica sono molto più facili da gestire.
dreamlax,

5
Lo pensavo anch'io, ma risulta che è troppo bello per essere vero. In questo caso devi conservare l'osservatore che restituisce addObserver e successivamente rimuoverlo, il che lo rende complicato quanto la creazione di un nuovo metodo, se non di più. Maggiori informazioni: toastmo.com/blog/2012/12/04/…
Andrew

42

se stai usando NSNotificationCenter per aggiornare la vista, non dimenticare di inviarlo dal thread principale chiamando dispatch_async:

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

1
è il post di notifica che deve verificarsi dal thread principale o solo quando si aggiorna effettivamente la vista, ovvero all'interno del metodo che riceve la notifica che si invia al thread principale?
Crashalot,

1
il thread da cui si invia la notifica è il thread che esegue le funzioni e quindi tenta di modificare l'interfaccia utente. puoi anche usare l'invio al thread principale all'interno delle funzioni, proprio come hai detto: D. dovrebbe avere lo stesso risultato, forse è ancora meglio: D
eiran,

1
@eiran, grazie mille fratello, ha funzionato solo dopo che ho scritto dentro dispatch_async
Arshad Shaik

2

SWIFT 5.1 della risposta selezionata per i neofiti

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

... altrove in un'altra classe ...

 func someMethod(){
        // All instances of TestClass will be notified
        NotificationCenter.default.post(name: "TestNotification", object: self)
 }
Utilizzando il nostro sito, riconosci di aver letto e compreso le nostre Informativa sui cookie e Informativa sulla privacy.
Licensed under cc by-sa 3.0 with attribution required.