ส่งและรับข้อความผ่าน NSNotificationCenter ใน Objective-C?


610

ฉันพยายามส่งและรับข้อความผ่านNSNotificationCenterใน Objective-C อย่างไรก็ตามฉันไม่สามารถหาตัวอย่างใด ๆ เกี่ยวกับวิธีการทำเช่นนี้ได้ คุณจะส่งและรับข้อความผ่านได้NSNotificationCenterอย่างไร


มีประโยชน์มากจริงๆขอบคุณ สิ่งหนึ่งที่วิธี addObserver ไม่ควรมีเครื่องหมายจุดคู่กึ่งท้ายหลังจากตัวเลือกที่ระบุ (อย่างน้อยก็ทำให้เกิดข้อยกเว้นในรุ่นนี้ของฉัน) ฉันพยายามแก้ไขรหัสด้านบน แต่ไม่ยอมรับการเปลี่ยนแปลงเนื่องจากปัญหาการจัดรูปแบบในรหัสต้นฉบับ
Braunius

3
นี่เยี่ยมมาก: cocoawithlove.com/2008/06/…
Aram Kocharyan

2
q นี้เป็นวิธีที่ธรรมดาเกินไปและกว้างเกินไป googleing เล็ก ๆ น้อย ๆ จะได้มีประโยชน์ที่ดี
Daij-Djan

นี่คล้ายกับคำถามที่เกี่ยวข้องที่นี่: stackoverflow.com/questions/7896646/…
David Douglas

55
ฉันพบว่ามันไร้สาระคำถามเช่นนี้ถูกปิดไม่สร้างสรรค์เมื่อผู้ใช้ของ Stack Overflow มีความเห็นอย่างชัดเจนถึงประโยชน์ของมัน
Chet

คำตอบ:


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

14
@Fulvio: ขึ้นอยู่กับว่าคุณได้รับหรือโพสต์การแจ้งเตือนที่อาจส่งผลต่อทุกส่วนของแอปพลิเคชันของคุณให้ใส่ไว้ใน AppDelegate ของคุณ หากคุณได้รับ / โพสต์การแจ้งเตือนที่มีผลต่อคลาสเดียวให้ใส่ไว้ในคลาสนั้นแทน
Dreamlax

1
@dreamlax Truth แต่มันก็คุ้มค่าที่สังเกตเพราะคำถามนี้ส่วนใหญ่จะถูกค้นหาโดย ios devs ใหม่ที่ทำให้ผู้ฟังการแจ้งเตือนมีชีวิตอยู่นานกว่าที่พวกเขาต้องการ ขณะนี้มีส่วนโค้งคุณมักจะไม่ใช้ dealloc และด้วยเหตุนี้บางคนอาจคิดว่าพวกเขาไม่ต้องปล่อยผู้ฟัง
Vive

7
อาจเป็นสิ่งที่ควรค่าแก่การกล่าวถึงว่า[super dealloc]ไม่อนุญาตให้ใช้การเรียกในวิธี dealloc-method ภายใต้ ARC; ส่วนที่เหลือดีทั้งหมด
tommys

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
ฉันสามารถเข้าถึงผู้สังเกตการณ์selfในเมธอด getTestNotification หรือไม่
ทำไม

ทำไมถึงใช่. getTestNotification เป็นวิธีการอินสแตนซ์และคุณสามารถเข้าถึงอินสแตนซ์ของตนเองผ่านตนเองภายใน
Michael Peterson

แค่นั้นแหละ. ฉันกำลังมองหาวิธีรับ UserInfo จากวิธีรับ
hasan

ดูเหมือนว่าความคิดของผู้สังเกตการณ์ทั้งหมดจะไม่ครอบคลุมทุกกรณี สิ่งนี้ไม่ทำงานเมื่อแอป ถูกปิดและแบบฟอร์มการแจ้งเตือนที่ศูนย์การแจ้งเตือนถูกแตะ วิธีการสังเกตการณ์ไม่ได้ถูกเรียก
Hasan

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

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 น่าจะมีผลเหมือนกัน, perheps มันจะดียิ่งขึ้น: D
eiran

1
@ eiran ขอบคุณมากพี่มันทำงานเฉพาะหลังจากที่ฉันเขียนไว้ใน dispatch_async
Arshad Shaik

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.