Swift 4 — проблема addObserver в Центре уведомлений

Я вылетаю и получаю ошибку unrecognized selector каждый раз, когда приходит Notification, и приложение пытается выполнить связанный с ним метод. Вот мой код, который находится в viewDidLoad:

let notificationCenter = NotificationCenter.default
notificationCenter.addObserver(self, selector: Selector(("sayHello")), name:NSNotification.Name(rawValue: "dataDownloadCompleted"), object: nil)

Метод sayHello() довольно прост — выглядит так:

func sayHello() {
    print("Hello")
}

Я убедился, что Notification успешно отправлено и доставлено успешно, так что проблема не в этом. Сбой происходит, когда приложение пытается действовать после прибытия Notification, выполняя метод sayHello(). Он продолжает выдавать мне эту ошибку unrecognized selector.

Любые идеи, что я делаю неправильно? (Кстати, это отлично работало со Swift 3 и Xcode 8, но теперь с Swift 4 и Xcode 9 синтаксис изменился [Xcode провел меня через необходимые исправления/обновления кода] - но сбои продолжают происходить.)


person Sirab33    schedule 11.10.2017    source источник
comment
Если бы вы использовали #selector, компилятор указал бы на проблему — sayHello должно быть @objc. Сравните stackoverflow.com/q/44390378/2976878   -  person Hamish    schedule 11.10.2017
comment
Используйте так #selector(yourVC.yourfunctionName)   -  person Mannopson    schedule 11.10.2017
comment
Как уже упоминалось @Hamish, вы должны использовать #selector(sayHello) и свою подпись метода, вы также должны передать объект уведомления (отбросить префикс NS) @objc func sayHello(_ notification: Notification)   -  person Leo Dabus    schedule 11.10.2017
comment
@Mannopson нет необходимости передавать префикс контроллера представления. Кстати, он всегда должен начинаться с заглавной буквы   -  person Leo Dabus    schedule 11.10.2017
comment
@LeoDabus Я всегда использую self вместо viewControllerName. Пример: #selector(self.functionName)   -  person Mannopson    schedule 11.10.2017
comment
себе это совершенно не нужно. Кстати, вы имеете в виду ViewControllerName, а не его экземпляр   -  person Leo Dabus    schedule 11.10.2017


Ответы (3)


Вы можете улучшить свой код, выполнив следующие действия:

extension Notification.Name {
    static let dataDownloadCompleted = Notification.Name(
       rawValue: "dataDownloadCompleted")
}

И используйте это так:

let notificationCenter = NotificationCenter.default
notificationCenter.addObserver(self,
                               selector: #selector(YourClass.sayHello),
                               name: .dataDownloadCompleted,
                               object: nil)

Но, как уже было сказано, проблема решается заменой на #selector

person Vladyslav Zavalykhatko    schedule 11.10.2017
comment
Имя не является участником уведомления - проблема возникает в Swift 4 - person Mohd Sadham; 18.01.2018

Swift 4.0 и Xcode 9.0+:

Отправить (разместить) уведомление:

NotificationCenter.default.post(name: Notification.Name("NotificationIdentifier"), object: nil)

OR

NotificationCenter.default.post(name: Notification.Name("NotificationIdentifier"), object: nil, userInfo: ["Renish":"Dadhaniya"])

Получить (получить) уведомление:

NotificationCenter.default.addObserver(self, selector: #selector(self.methodOfReceivedNotification(notification:)), name: Notification.Name("NotificationIdentifier"), object: nil)

Обработчик функции-метода для полученного уведомления:


@objc func methodOfReceivedNotification(notification: Notification) {}

Swift 3.0 и Xcode 8.0+:

Отправить (разместить) уведомление:


NotificationCenter.default.post(name: Notification.Name("NotificationIdentifier"), object: nil)

Получить (получить) уведомление:


NotificationCenter.default.addObserver(self, selector: #selector(YourClassName.methodOfReceivedNotification(notification:)), name: Notification.Name("NotificationIdentifier"), object: nil)

Обработчик метода для полученного уведомления:

func methodOfReceivedNotification(notification: Notification) {
  // Take Action on Notification
}

Удалить уведомление:

deinit {
  NotificationCenter.default.removeObserver(self, name: Notification.Name("NotificationIdentifier"), object: nil)
}

Swift 2.3 и Xcode 7:

Отправить (опубликовать) уведомление

NSNotificationCenter.defaultCenter().postNotificationName("NotificationIdentifier", object: nil)

Получить (получить) уведомление


NSNotificationCenter.defaultCenter().addObserver(self, selector: #selector(YourClassName.methodOfReceivedNotification(_:)), name:"NotificationIdentifier", object: nil)

Обработчик метода для полученного уведомления

func methodOfReceivedNotification(notification: NSNotification){
  // Take Action on Notification
}

Ссылка: https://medium.com/@javedmultani16/notification-in-swift-4-8b0db631f49d

person Mr.Javed Multani    schedule 24.08.2019

person    schedule
comment
если вы делаете свой код таким образом, наблюдатель удаляется, когда представление исчезает (например, когда другое представление помещается в стек навигации), но если это новое представление возвращается к исходному представлению (где раньше жил наблюдатель), есть нет наблюдателя уведомлений. Лучше поставить addObserver в viewWillAppear. - person Michael Dautermann; 03.07.2018